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前 言 


编写 目的 与 背景 
众所周知 ， 当 前 社会 需求 和 高 校 课程 设置 严重 脱节 ， 一 方面 企业 找 不 到 可 迅速 上 手 的 人 才 ， 另 一 


方面 大 学 生 就 业 难 。 如 果 有 一 些 面向 工作 应 用 的 案例 参考 书 ， 让 大 学 生得 以 参考 ， 并 能 亲手 去 做 ， 势 
必 能 缓解 这 种 矛盾 。 本 书 就 是 这 样 一 本 书 :项 目 开发 案例 型 的 、 面 向 工作 应 用 的 软件 开发 类 图 书 。 编 


发 学 习 。 


第 四 ， 从 书 第 1 版 于 2008 年 6 月 出 版 ， 于 2011 年 1 月 改版 ， 因 为 编写 细腻 ， 配 备 全 程 视频 备 受 
读者 瞩目 ， 丛 书 累计 销售 16 万 册 ， 成 为 近年 来 最 受 欢迎 的 软件 开发 项 目 案例 类 从 书 之 一 。 


在 以 上 背景 下 ， 我 们 根据 读者 朋友 的 反馈 ， 与 时 俱 进 ， 对 从 书 进行 了 改版 。 
本 书 特点 


铬 m 视频 讲解 


对 于 初学 者 来 说 ， 视 频 讲解 是 最 好 的 导师 ， 它 能 够 引导 初学 者 快速 入 门 ， 使 初学 者 感受 到 编程 的 
快乐 和 成 就 感 ， 增 强 进一步 学 习 的 信心 。 鉴 于 此 ， 本 书 为 每 一 个 案例 都 配备 了 视频 讲解 ， 初 学 者 可 以 
通过 视频 讲解 实现 案例 中 的 功能 。 


佬 典型 案例 


本 书 案例 均 从 实际 应 用 角度 出 发 ， 应 用 了 当前 流行 的 技术 ， 涉 及 的 知识 广泛 ， 读 者 可 以 从 每 个 案 
例 中 积累 丰富 的 实战 经 验 。 


所 代码 注释 
为 了 便于 读者 阅读 程序 代码 ， 书 中 的 代码 均 提 供 了 详细 的 注释 ， 并 且 整 齐 地 纵向 排列 ， 可 使 读者 
快速 领略 作者 意图 。 


难 驹 驹 ”PHP 典型 模块 开发 全 程 实录 


的 代码 贴 士 

案例 类 书籍 通常 会 包含 大 量 的 程序 代码 ， 宛 长 的 代码 往往 令 初 学 者 望 而 生 旦 。 为 了 方便 读者 阅读 
和 理解 代码 ， 本 书 避 免 了 连续 大 篇 幅 的 代码 ， 将 其 分 割 为 多 个 部 分 ， 并 对 重要 的 变量 、 方 法 和 知识 点 
设计 了 独 具 特 色 的 代码 贴 士 。 


他 知识 扩展 


为 了 增加 读者 的 编程 经 验 和 技巧 ， 书 中 每 个 案例 都 标记 有 注意 、 技 巧 等 提示 信息 ， 并 且 在 每 章 中 
都 提供 有 一 项 专题 技术 。 


本 书 约定 


由 于 篇 幅 有 限 ， 本 书 每 章 并 不 能 逐一 介绍 案例 中 的 各 模块 。 作 者 选择 了 基础 和 典型 的 模块 进行 介 
绍 ， 对 于 功能 重复 的 模块 ， 由 于 技术 、 设 计 思 路 和 实现 过 程 基本 雷同 ， 因 此 没有 在 书 中 体现 。 读 者 在 
学 习 过 程 中 若 有 相关 疑问 ， 请 登录 本 书 官方 网 站 。 本 书 中 涉及 的 功能 模块 在 光盘 中 都 附带 有 视频 录像 ， 
方便 读者 学 习 。 


适合 读者 


本 书 适合 作为 计算 机 相关 专业 的 大 学 生 、 软 件 开发 相关 求职 者 和 爱好 者 的 毕业 设计 和 项 目 开 发 的 
参考 书 。 
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1.63 
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目录 


1.6.6 PHP 中 操作 Cookie 技术 .………… 
1.6.7 在 JavaScript 中 操作 Cookie 技术 .. 
1.6.8 ”用户 自动 登录 技术 
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2.2 简单 数字 计数 器 ... 
2.3 图 形 数 字 计 数 器 
2.4 GD2 图 形 计数 器 .. 
2.5 数据库 数 字 计 数 器 ... 
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2.6.1 通过 数据 库 记 录 网 站 访问 量 .. 
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3.1.1 模块 概述 . 
3.1.2 ”功能 结构 . 


3.1.3 ”程序 预览 . 
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ER A 58 
3.2.2 ”创建 数据 表 
3.2.3 ”数据库 连 接 文件 
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3.4.3 ”为 用 户 划分 个 人 空间 文件 目录 .… 
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有 84 
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注册 登录 模块 


(PHP+Ajax+Zend_Mail 实现 ) 


从 动态 网 站 开始 ， 注 册 登 录 模 块 就 成 为 了 网 站 必 不 可 少 的 组 成 部 
分 ， 因 为 其 功能 单一 ， 实 现 相 对 简单 ， 从 而 成 为 入 门 级 的 模块 。 随 着 
Web 2.0 的 发 展 ,， 人 机 交互 成 为 主流 趋势 ， 注册 登录 模块 也 顺应 了 时 代 
的 发 展 ， 变 得 更 加 有 “亲和力 ”。 像 过 去 那 种 花 了 许久 时 间 填 写 注 册 表 
单 ， 提交 后 发 现 信息 填写 错误 ,需要 重新 填写 的 惨痛 经 历 越 来 越 少 。 随 
着 Ajax 技术 的 兴起 ， 这 种 情况 就 更 少 发 生 了 。 本 章 将 开发 一 个 功能 齐 
全 并 有 着 良好 交互 性 的 注册 登录 模块 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

让 注册 信息 实时 提示 
E-mail 激活 技术 
GD2 验证 码 
更 换 验 证 码 
防 SQL 注入 技术 


各 吾 吾 至 


PHP 典型 模块 开发 全 程 实录 


1.1 注册 及 登录 模块 概述 


注册 及 登录 模块 的 功能 比较 单一 ， 除 了 实现 验证 用 户 信息 的 有 效 性 和 一 些 必要 的 安全 设置 外 ， 更 
多 的 是 如 何 提高 程序 的 可 操作 性 ， 更 好 地 方便 用 户 。 注 册 及 登录 模块 由 用 户 注册 、 用 户 登 录 和 找 回 密 
码 3 部 分 组 成 。 


1.1.1 用 户 注册 流程 


用 户 注册 的 信息 分 为 必 填 信 息 和 可 选 信息 。 必 填 信息 默认 是 可 见 的 ， 而 可 选 信息 默认 是 隐藏 的 ， 
在 必 填 信息 不 完整 时 ，“ 注 册 ” 按 钮 是 不 可 用 的 。 对 于 可 选 信息 ， 用 户 可 以 选择 填写 或 不 填写 。 但 如 
果 填 写 了 ， 那 么 就 需要 对 信息 的 正确 性 进行 检验 。 当 用 户 注册 成 功 后 ， 系 统 将 向 用 户 所 填写 的 E-mail 
发 送 一 封 激活 邮件 ， 用 户 只 有 激活 后 ， 才 可 使 用 该 账号 进行 登录 。 用 户 注册 流程 图 如 图 1.1 所 示 。 


< > 


[ 必 填 信息 _ 
入 用 | 输入 信息 


器 | 元 评注 册 


-组 是 1 才 瑟 三思 | 


提交 


图 1.1 用 户 注册 流程 图 
1.1.2 用 户 登 录 流 程 
用 户 登 录 时 ， 除 了 用 户 名 和 密码 外 ， 还 需要 填写 随机 生成 的 验证 码 。 当 验证 码 显示 不 清楚 时 ， 可 
以 通过 超 链接 进行 更 换 而 无 须 重新 刷新 页 面 。 登 录 处 理 对 登录 进行 了 一 些 限制 ， 例 如 ， 输 入 3 次 无 效 


用 户 名 后 ， 将 使 用 Cookie 技术 禁止 该 用 户 继续 登录 ; 连续 3 次 输入 错误 密码 后 ， 该 账号 将 被 冻结 。 用 
户 登 录 流 程 图 如 图 1.2 所 示 。 
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更 换 验证 码 |4- 一 了 输入 信息 


图 1.2 用 户 登录 流程 图 


1.1.3” 找 回 密码 流程 


当 用 户 因 为 个 别 原因 坊 记 密码 时 ， 可 以 通过 密码 保护 问题 及 答案 来 找 回 密码 。 如 果 用 户 输入 正确 ， 
那么 系统 将 密码 通过 电子 邮件 的 形式 发 到 用 户 的 注册 邮箱 中 。 如 果 没 有 填写 密码 保护 的 问题 及 答案 或 


者 回答 错误 ， 则 无 法 找 回 。 找 回 密码 流程 图 如 图 1.3 所 示 。 
找 回 窗 码 


图 1.3 找 回 密码 流程 图 


1.1.4 程序 预览 


注册 登录 模块 主要 由 两 个 界面 组 成 ， 即 注册 页 面 和 登录 页 面 。 注 册页 面 与 登录 页 面 的 运 和 
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1.4 和 图 1.5 所 示 。 


‘ 


BE CD saannn 只 明日 主 这 / 纳入 于 / 的 得 伯 验 


i TR a RT FN SAN BA 


用 户 营 理 


加 果 你 没有 注册 ,请 您 注册 后 再 和 登录。 如 果 炊 已 经 注册 ,但 忘记 了 密码 ,您 可 以 通过 
找 同 密码 将 您 的 密码 找 回 , 


注册 名 称 : PP 用 户 各 被 占用 ! 
[eeee。 证 可行 全 要求。 证 到 朗 : 弹 
确 


rzoft88888cohu le 


密 保 问题 : | 困 日 科技 
密 保 答案 。 图 书 开发 


育 实 姓名 : | 轩 日 科技 

出 生日 期 : |1999-12-01 用 户 
联系 电话 : D031-B497777 | 
QQ 号 码 :2465827366 


技术 服务 热线 : 400-875-1066 传真 : 0431-84939TTT 例 业 邮箱 : nrbe eanrbccd con 
公司 地 址 : 长 碍 市 亚 事 广场 [ 座 ”六 TCF 备 125869 
Copyritht Or arbecd con 人 1 Rights Reservedl 


图 1.4 注册 页 面 的 运行 效果 


‘$ 


BE CD aannn 只 明日 主 站 / 加 各 交 源 
a Br ART aN 


用 户 管理 


加 果 您 没有 注册 ， 请 您 注册 后 再 登录 。 如 果 悠 已 经 注册 ,但 忘记 了 密码 ,您 可 以 通过 
找 回 密码 将 您 的 密码 找 回 。 


[项 | 5650 ET 请 


验证 玛 : 
开本 | 到 本 


技术 服务 热 如 : 400-675-1066 传真 : 0431-84939777 企业 邮箱 : arbecdpwrbeed con 
公司 地 址 : 长 春 市 王 泰 广场 C 座 吉 ICP 备 125689 
Copyrieht Oem mrbecd com ALl Rights Reservedl 


图 1.5 登录 页 面 的 运行 效果 


1.2 数据库 设计 


本 模块 应 用 MySQL 数据 库存 储 用 户 注册 信息 , 有 关 MySQL 数据 库 及 数据 表 的 创建 将 在 本 节 进 行 


@ 
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详细 讲解 。 
1.2.1 数据 库 设 计 概 述 


注册 及 登录 验证 模块 使 用 的 数据 库 是 db_reglog， 库 中 包含 了 一 个 数据 表 tb_member， 该 表 中 有 12 
个 字段 。tb_member 表 结 构 及 说 明 如 图 1.6 所 示 。 
中 服务 器 : localhost ， 成 数据 库 : db_reglog ， 国 表 : tb_member 
赎 剂 览 ” 因 结构 “如 SQL ”万 搜 索 也 疾 入 。 茵 导出 “了 虎 Import 笑 操 作 。 曾 清空 “网 副 除 


字段 类 型 整理 属 Null 默 额外 说 明 
性 认 

| int(4) 否 auto_increment 。 自动 编号 
厂 name varchar(100) gb2312_chinese_ci 否 用 户 名 称 
mm password varchar(200) gb2312-chinese-ci 否 用 户 密 码 
厂 answer varchar(100) gb2312_chinese_ci 否 密 保 答案 
厂 question varchar(100) gb2312_chinese_ci 否 密 保 问题 
厂 email varchar(100) gb2312_chinese_ci 否 Email 
厂 realname varchar(200) gb2312_chinese_ci 否 车 录 限 制 
厂 birthday dare 否 出 生日 其 
厂 telephone varchar(20) gb2312_chinese_ci 否 电话 号 码 
F qq varchar(15) gb2312_chinese_ci 否 9 虽 
厂 count int0) 否 0 里 录 限 制 
厂 active Inc0) 否 0 是 否 激活 


图 1.6 tb_member 表 结 构 及 说 明 
1.2.2 ”封装 数据 库 连 接 - 操 作 类 


为 了 便于 维护 ， 减 少 代码 元 余 ， 本 模块 使 用 了 一 个 简单 的 自 定义 数据 库 类 。 下 面 给 出 类 中 的 主要 
代码 。 

(1) 类 中 的 成 员 变 量 。 其 中 $host、S$name、S$pwd 和 $dBase 变量 是 类 初始 化 时 所 使 用 的 ， 有 默认 
值 ， 其 他 变量 都 是 相关 的 数据 信息 。 代 码 如 下 : 


<?php 

class opmysql{ 

private $host = \ocalhost'; /服务 器 地 址 
private $name = 'root'; /登录 账号 
private $pwd = "111°; /登录 密码 
private $dBase = 'db_reglog'; /数据库 名 称 
private $conn =" /数据 库 连接 资源 
private $result = /| 结果 集 

private $msg = "; /| 返回 结果 
private $fields; /| 返回 字段 
private $fieldsNum = 0; /1 返回 字段 数 
private $rowsNum = 0; /| 返回 结果 数 
private $rowsRst = "; /1 返回 单条 记录 的 字段 数组 
private $filesArray = array(); // 返 回 字段 数组 
private $rowsArray = array(); // 返 回 结果 数组 


》 PHP 典型 模块 开发 全 程 实录 


(2) 构造 函数 。 该 构造 函数 有 4 个 参数 ， 这 4 个 参数 都 可 以 被 省 略 而 使 用 默认 值 。 参 数 处 理 后 ， 
调用 init_conn0) 函 数 。 代 码 如 下 : 


function __construct($host=",$name=",$pwd=",$dBase="){ 
if($host !=") 

S$this->host = $host; 

if($name != ") 

S$this->name = $name; 

if($pwd {= ") 

S$this->pwd = $pwd; 

if($dBase !=") 

S$this->dBase = $dBase; 

S$this->init_conn(); 


init_conn() 函 数 根据 成 员 变 量 中 的 值 来 创建 数据 库 连接 源 。 代 码 如 下 : 


// 连 接 数据 库 

function init_conn(){ 
$this->conn=@mysql_connect($this->host,$this->name,$this->pwd); 
@mysql_select_db($this->dBase, $this->conn); 

mysql_query("set names gb2312"); // 设 置 编码 


2. 


(3) 类 中 的 操作 函数 。 
@ 查询 函数 。 查 询 函 数 根据 传 过 来 的 SQL 语句 进行 查询 ， 并 将 查询 结果 保存 到 成 员 变 量 $result 
中 。 符 号 “@” 的 作用 是 屏蔽 错误 信息 。 代 码 如 下 : 


// 查 询 结果 

function mysql_query_rst($sql){ 
if($this->conn == "}{ 
$this->init_conn(); 


让 
$this->result = @mysql_query($sql, $this->conn); 
} 


@ 返回 查询 记录 数 函 数 。 该 函数 根据 查询 结果 返回 记录 数 。 代 码 如 下 : 


function getRowsNum($sql){ 
$this->mysql_query_rst($sql); 
if(mysql_errno() == 0){ 

return @mysql_num_rows($this->result); 
}else{ 

return "; 


@ 取得 记录 数组 函数 。 该 函数 将 查询 结果 输出 成 一 个 数组 并 返回 , 其 处 理 的 是 单条 记录 。 代码 如 下 : 


第 1 章 注册 登录 模块 (PHP+Ajax+Zend_ Mail 实现 ) 


function getRowsRst($sqlX{ 

$this->mysql_query_rst($sql); 

if(mysql_error() == 0){ 

$this->rowsRst = mysql_fetch_array($this->result, MYSQL_ASSOC); 
return $this->rowsRst; 

}else{ 

return "; 

} 

. 


@ 取得 记录 数组 函数 。 功 能 同上 ， 只 是 这 里 返回 一 个 有 多 条 记录 的 二 维 数组 。 代 码 如 下 : 


// 取 得 记录 数组 (多 条 记录 ) 
function getRowsArray($sql){ 
S$this->mysql_query_rst($sql); 
if(mysql_errno() == 0){ 
while($row = mysql_fetch_array($this->result, MYSQL_ASSOC)) { 
S$this->rowsArray[] = $row; 

} 

return $this->rowsArray; 
}else{ 

return "; 

上 


@@ 返回 更 新 、 删 除 、 添 加 的 记录 数 函 数 。 该 函数 用 来 更 新 、 删 除 、 添 加 记录 并 获取 受 影 响 的 记录 
用 来 判断 操作 是 否 成 功 。 代 码 如 下 : 
function uidRst($sql}{ 


if($this->conn == "}{ 
$this->init_conn(); 


} 

@mysql_query($sql); 

$this->rowsNum = @mysql_affected_rows(); 
if(mysql_errno() == 0){ 

return $this->rowsNum; 

}else{ 

return "; 
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@ 释放 结果 集 函 数 。 该 函数 用 来 将 不 再 使 用 的 数据 删除 ， 释 放 内 存 。 代 码 如 下 : 


function close_rst(X{ 
mysql_free_result($this->result); 
S$this->msg = "; 
S$this->fieldsNum = 0; 
S$this->rowsNum = 0; 
S$this->filesArray = "; 
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Sthis->rowsArray = "; 
} 


@ 关闭 数据 库 函 数 。 代 码 如 下 : 


/关闭 数据 库 

function close_conn(){ 
S$this->close_rst(); 
mysql_close($this->conn); 


Sthis->conn ="; 

} 

} 

$conne = new opmysql(); 
一 人 人 人 


1.3 用 户 注 册 


1.3.1 用 户 注册 功能 概述 


在 用 户 注册 时 ， 需 要 填写 的 必 填 信息 包括 用 户 名 、 密 码 和 E-mail。 只 有 这 些 必 填 信息 填写 完整 并 
正确 时 ，“ 注 册 ” 按 钮 才 被 激活 。 这 时 用 户 可 以 选择 注册 或 填写 详细 信息 后 再 进行 注册 。 在 整个 注册 
的 过 程 中 ， 用 户 的 每 一 次 操作 ， 系 统 都 将 给 出 友好 的 提示 来 帮助 用 户 完成 注册 。 注 册页 面 的 运行 结果 
如 图 1.7 所 示 。 


BE CD aang 只 习 日 三 站 1 编程 次 源 / 加 各 人 法 
Ca 


L 产品 介绍 免费 下 载 。 ”在 拷 订 冯 。” 技 玉 支 府 。 ”服务 论 寺 


加 果 您 没有 注册 ,请 您 注册 后 再 芝 录 - 加 果 您 已 经 注册 ,但 亡 记 了 赣 码 ,次 可 以 通过 
找 回 密码 将 您 的 密码 拒 回 . 


用 户 名 被 占用 ! 

市 码 符合 要 求 。 密 码 齐 度 : 对 
: 证 码 给 入 下 确 

电子 邮 宿 。 rs0ft88888cohu con 输入 正确 


密 保 问 是 。 胃 百 科技 
密 保 答案 :图书 开发 

真实 姓名 : | 胃 日 科技 

出 生日 期 : |1959-12-01 

联系 电话 : |0431-849777T 
346827366 


技术 服务 热 厂 : 400-6Ts-1063 传真 : 0431-84939777 全 JH 邮箱 :arbe clanrtced =on 
公司 地 址 : 长 春 币 至 雪 广 声 c 话 ” 吉 ICP 苗 lessae 
Cooyridht Der arbecd eon AL Lights Reseryeal 


1.7 注册 页 面 的 运行 结果 
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用 户 注册 功能 包含 5 个 核心 文件 ， 分 别 是 register.php〔〈 注 册 表 单 页 ) 、register_chk.php〈 注 册 处 理 
页 ) 、chkname.php《〈 检 测 用 户 名 ) 、registerjs〈 注 册 信 息 验 证 脚本 ) 和 activation.php (激活 处 理 页 ) 。 


1.3.2 ”注册 页 设计 


注册 表单 页 (register.php) 中 , 主要 包含 填写 信息 的 文本 框 、 信 息 提 示 的 div 标签 和 3 个 相关 按钮 。 
注册 页 面 中 使 用 的 元 素 如 表 1.1 所 示 。 


表 1.1 注册 页 面 中 使 用 的 元 素 


元 素 名 称 元 素 属性 值 说 了 明 
script "js/xmlhttprequest.js" 载 入 xmlHttpRequest 对 象 脚本 
script "js/register.js" 载 入 注册 信息 验证 脚本 
text id="regname" name="regname" 注册 名 称 
div id="namediv" 注册 名 称 提示 信息 
password id="regpwdl" name="regpwdl" 注册 密码 
div id="pwddiv1" 密码 提示 信息 
password id="regpwd2" name="regpwd2" 确认 密码 
div id="pwddiv2" 确认 密码 提示 信息 
text id="email" name="email” 电子 邮箱 
div id="emaildiv" E-mail 提示 信息 
div id="morediv" style="display:none;" 隐藏 /显示 非 必 填 信息 
text id=" guestion" name="guestion" 密码 保护 问题 
text id="answer" name="answer" 密码 保护 答案 
text id="realname" name="realname" 真实 姓名 
text id="birthday" name="birthday" 出 生日 期 
text id="telephone" name="telephone" 联系 电话 
text id="qq "name=" qq" QQ 号码 
button id="regbtn" disabled="disabled" “注册 ”按钮 
button id="morebtn" 显示 /隐藏 全 部 注册 信息 
button id="logbtn" 返回 登录 页 面 
div id="imgdiv" style=" visibility: hidden;" 等 待 进度 条 


当 用 户 向 注册 页 面 中 的 表单 元 素 输入 信息 时 ， 将 会 触发 信息 处 理 脚 本 registerjs。 下 面 逐 行 讲解 


register.js 脚本 文件 。 


(1) 在 registerjs 中 ， 首 先 写 了 一 个 通用 函数 ， 来 避免 重复 输入 document.getElementByld(…')。 


这 个 函数 的 意思 是 返 


function $(idValueX{ 
retum document.getElementByld(idValue); 


} 


回 被 触发 的 “id=idValue” 元 素 对 象 。 


@ 
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(2) window.onload 事件 ， 表 示 当 窗 体 被 载 入 时 触发 。function(){…} 表 示 当 页 面 被 载 入 时 所 要 进 
行 的 操作 。 


window.onload = function(){ 


} 


(3) 下 面 所 有 的 代码 都 是 上 面 的 函数 。 
@ 将 焦点 定位 到 注册 名 称 文本 框 ， 方 便 用 户 操作 。 接 下 来 声明 5 个 变量 ， 这 5 个 变量 代表 5 项 要 
检测 的 数据 结果 。 当 检测 数据 为 “合格 ”时 ， 代 表 它 的 变量 将 等 于 yes。 代 码 如 下 : 
$(regname')focus(); /焦点 定位 到 注册 名 称 文本 框 


/声明 5 个 变量 ， 表 示 要 检测 的 5 项 数据 
var cname1,cname2,cpwd1,cpwd2,cemail; 


@ chkreg() 函 数 是 每 一 次 触发 键盘 事件 后 都 要 调用 的 。 该 函数 判断 5 个 变量 的 值 ， 只 有 当 所 有 变 
量 都 等 于 yes 时 ，“ 注 册 ” 按 钮 才 会 被 激活 。 代 码 如 下 : 


function chkreg(){ 

if((cname1 == 'yes') && (cname2 == "yes') && (cpwd1 == "yes') && (cpwd2 == 'yes') && (cemail == 'yes')}{ 
$(regbtn').disabled = false; 

Jelse{ 

$(regbtn').disabled = true; 

} 

; 


@ 验证 用 户 名 。 当 用 户 输入 注册 名 称 时 ， 该 函数 会 把 用 户 的 每 次 输入 都 做 一 下 正则 判断 ， 并 根据 
结果 设置 不 同 的 cnamel 值 。 代 码 如 下 : 


// 验 证 用 户 名 

$(‘regname').onkeyup = function (X{ 

1/ 获取 注册 名 称 

name = $(regname').value; 

cname2 ="; 

if(name.match(/^[a-zA-Z_]*/) =="X 

$('namediv').innerHTML = '<font color=red> 必 须 以 字母 或 下 划 线 开头 </font>'; 
cname1 = "; 

}else if(name.length <= 3){ 

$('namediv').innerHTML = '<font color=red> 注 册 名 称 必须 大 于 3 位 </font>'; 
cname1 ="; 


}else{ 
$(namediv).innerHTML = '<font color=green> 注 册 名 称 符合 标准 </font>'; 


cname1 = yes'; 
有 
chkreg(); 
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@ 当 注 册 名 称 文本 框 失 去 焦点 时 ， 即 用 户 输入 完毕 转 到 页 面 中 其 他 元 素 时 ， 将 检测 用 户 名 是 否 重 
复 。 用 户 名 判断 使 用 Ajax 技术 调用 了 chkname.php 页 (chkname.php 页 的 代码 将 在 1.3.3 节 给 出 ) 并 根 
据 chkname.php 的 返回 值 ， 在 div 标签 中 显示 判断 结果 。 代 码 如 下 : 


// 当 用 户 名 文本 框 失去 焦点 时 ， 系 统 调用 处 理 页 查看 是 否 存在 该 用 户 
$('regname').onblur = function(X{ 

name = $('regname').value; /| 获取 注册 名 称 
// 当 用 户 名 称 的 格式 输入 合格 后 ， 才 进行 这 一 步 操作 

if(cname1 == yes'){ 

xmlhttp.open(get,chkname.php?name='+name,true); // 创 建新 请 求 
xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readyState == 4){ 

if(xmlhttp.status == 200){ 

var msg = xmlhttp.responseText' /获取 响应 页 内 容 
if(msg == "1'){ 

$(‘namediv').innerHTML="<font color=green> 恭 喜 您 ， 该 用 户 名 可 以 使 用 !</font>"; 
cname2 = yes'; 

}else if(msg == '2){ 

$(namediv').innerHTML="<font color=red> 用 户 名 被 占用 !</font>"; 

cname2 ="; 

}else{ 

$('namediv').innerHTML="<font color=red>"+msg+"</font>"; 

cname2 ="; 

} 

} 
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chkreg(); /检查 是 否 要 激活 “注册 ”按钮 


xmlhttp.send(null); 


@@ 验证 密码 。 验 证 密码 时 ， 除 了 可 以 限制 密码 的 长 度 外 ， 还 可 以 判断 密码 的 强度 。 例 如 下 面 的 代 
码 中 ，6 一 12 位 之 间 的 密码 都 是 弱 ， 如 果 大 于 12， 但 密码 单一 ， 如 都 是 数字 或 文字 ， 那 么 密码 强度 就 
为 中 ， 否 则 强度 就 为 高 。 代 码 如 下 : 


// 验 证 密码 

$('regpwd1').onkeyup = function(){ 

pwd = $('regpwd1').value; 

pwd2 = $('regpwd2').value; 

if(pwd.length < 6){ 

$(‘pwddiv1').innerHTML = '<font color=red> 密 码 长 度 最 少 需要 6 位 </font>'; 

cpwd1 ="; 

}else if(pwd.length >= 6 && pwd.length < 12){ 

$(‘pwddiv1').innerHTML = '<font color=green> 密 码 符合 要 求 。 密 码 强度 : 弱 </font>'; 
cpwd1 = 'yes'; 
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}else if((pwd.match(/^[0-9]*$)!=null) || (pwd.match(/*[a-zA-2Z]*$/) != null ))f 
$(‘pwddiv1').innerHTML = '<font color=green> 密 码 符合 要 求 。 密 码 强度 : 中 </font>'; 
cpwd1 = yes'; 
}else{ 
$(pwddiv1').innerHTML = '<font color=green> 密 码 符合 要 求 。 密 码 强度 : 高 </font>'; 
cpwd1 = yes'; 
; 
if(pwd2 {= " &8& pwd != pwd2){ 
$('pwddiv2').innerHTML = '<font color=red> 两 次 密码 不 一 致 !</font>'; 
cpwd2 ="; 
}else if(pwd2 !=" && pwd == pwd2){ 
$('pwddiv2').innerHTML = '<font color=green> 密 码 输入 正确 </font>'; 
cpwd2 = yes'; 
这 
chkreg(); 
} 


@ 二 次 密码 的 验证 比较 简单 ， 只 要 第 一 、 第 二 次 密码 相等 就 可 以 。 代 码 如 下 : 


// 验 证 二 次 密码 

$('regpwd2').onkeyup = function(){ 

pwd1 = $('regpwd1').value; 

pwd2 = $('regpwd2').value; 

if(pwd1 != pwd2X 

$(pwddiv2').innerHTML = '<font color=red> 两 次 密码 不 一 致 !</font>'; 
cpwd2 = "; 

}else{ 

$(‘pwddiv2').innerHTML = '<font color=green> 密 码 输入 正确 </font>'; 
cpwd2 = yes'; 

chkreg(); 

} 

上 


@ E-mail 验证 主要 是 对 E-mail 的 格式 进行 检查 ， 输 入 的 字 串 中 必须 包含 “@” 和 “.”， 同 时 ， 
这 两 个 符号 的 位 置 既 不 能 在 首 和 尾 ， 也 不 能 连 在 一 起 。 代 码 如 下 : 


/验证 E-mail 

S$(‘email').onkeyup = function(X{ 

emailreg = /MWw+([-+.]W+)*@\w+([-.J\Ww+)\\w+ ([-.N\Ww+)*$/; 
$('email').value.match(emailreg); 

if($(email).value.match(emailreg) == nullX{ 

$(‘emaildiv').innerHTML = '<font color=red> 错 误 的 E-mail 格式 </font>'; 
cemail = "; 

}else{ 

$(‘emaildiv').innerHTML = '<font color=green> 输 入 正确 </font>'; 

cemail = yes'; 
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chkreg(); 


上 面 的 部 分 是 对 必 填 项 的 检查 。 如 果 用 户 希望 填写 更 详细 的 资料 ， 可 以 单 击 “ 详 细 资料 ”按钮 。 
代码 如 下 : 


// 显 示 / 隐 藏 详细 信息 

$('morebtn').onclick = function(X{ 

if($('morediv').style.display == "){ 
S$('morediv').style.display = "none'; 

}else{ 

$('morediv').style.display = "; 

} 

3 


@ 所 有 信息 填写 完毕 并 通过 验证 后 ， 就 可 以 注册 了 。 用 户 单 击 “ 注 册 ” 按 钮 时 ， 该 段 代码 应 用 
Ajax 技术 将 获取 的 用 户 信息 以 URL 的 形式 传 给 register_chk.php 页 ， 并 根据 返回 结果 做 出 相应 的 处 理 。 
代码 如 下 : 


/正式 注册 

$(regbtn').onclick = function(){ 

$(imgdiv').style.visibility = visible'; 

url = 'register_chk.php?name='+$('regname’).value+'&pwd='+$('regpwd1').value+'&email='+$('email').value; 
url += '&question=" +$(question ).value+'&answer='+$(answer).value; 
Url += '&realname=' +$('realname').value+'&birthday="+$('birthday').value; 
url += '&telephone='+$(telephone').value+'&qq='+$(qq').value; 
alert(url); 

return false; 

xmlhttp.open(get ,urltrue); 

xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readyState == 4){ 

if(xmlhttp.status == 200){ 

msg = xmlhttp.responseText; 

if(msg == "1°){ 

alert(' 注 册 成 功 ， 请 到 您 的 邮箱 中 获取 激活 码 ! '); 

location='main.php'; 

}else if(msg == -1 

alert(' 您 的 服务 器 不 支持 POP3， 请 仔细 检查 ! ! 小 

}else{ 

alert(msg); 

让 

| 

S$(imgdiv').style.visibility = 'hidden'; 

党 


xmlhttp.send(null); 
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1.3.3 ”检测 用 户 名 是 否 被 占用 


检测 用 户 名 是 否 被 占用 使 用 的 是 chkname.php 文件 。 其 作用 是 检查 用 户 名 是 否 重 复 ， 并 且 返 回 判 
断 结果 。 该 页 的 代码 如 下 : 


<?php 

// 载 入 数据 库 类 

include_once "conn/conn.php"; 

// 根 据 name 查询 tb_member 表 

$sql = "select * from tb_member where name=".$_GET[name']."™"; 


// 获 取 查 询 结 果 的 记录 数 

$num = $conne->getRowsNum($sql); 

if($num == 1X // 如 果 有 记录 
echo '2'; 

}else if($num == 0X{ // 如 果 没 有 记录 
echo't'; 

}else{ 

echo $conne->msg_error(); /否则 出 错 

和 


1.3.4 注册 信息 处 理 


在 注册 信息 处 理 页 (register_chk.php〉 中， 将 JavaScript 传递 过 来 的 值 连 成 insert 语句 并 执行 。 如 
果 添 加 成 功 ， 则 使 用 邮件 函数 发 送 一 封 激活 邮件 ， 并 弹出 提示 框 ， 结 果 如 图 1.8 所 示 。 


XxX | 


| 


图 1.8 注册 成 功 提示 框 


在 处 理 代码 中 包含 两 种 邮件 的 发 送 方法 .一 种 是 使 用 局 域 网 ， 向 指定 SMTP 服务 器 的 邮箱 中 发 送 邮 
件 ， 其 中 的 cym3100@163.com 邮箱 是 在 SMTP 服务 器 中 注册 的 测试 邮箱 ; 另 一 种 适用 于 本 地 网 络 ， 通 
过 指定 的 邮箱 (mrsoft@mrsoft.com 是 在 WinMail 软件 中 注册 的 邮箱 ， 用 户 名 是 mrsof@mrsoftcom， 密 
码 是 111) 向 用 户 提供 的 有 效 邮 箱 中 发 送 邮件 。 处 理 页 代码 如 下 : 


<?php 
include_once 'conm/conn.php'; 
require_once 'Zend/Mail.php'; // 调 用 发 送 邮件 的 文件 
require_once 'Zend/Mail/Transport/Smtp.php’; // 调 用 SMTP 验证 文件 


$reback = '0'; 
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$url = 'http://.$_SERVER['SERVER_NAME'. :82'.dimame($_SERVER['SCRIPT_NAME])./activation.php'; 
$url .= ?name='trim($_GET[name]).&pwd='.md5(trim($_GET[pwd])); 

// 发 送 激活 邮件 

$subject=" 激 活 码 的 获取 "; 

$mailbody=' 注 册 成 功 。 您 的 激活 码 是 : 

"<a href=".$url." target=”blank">'.$url.'</a><br>'…' 请 点 击 该 地 址 ， 激 活 您 的 用 户 ! '; 

// 定 义 邮 件 内 容 

$envelope=" mrsoft@mrsoft.com "; // 定 义 登录 使 用 的 邮箱 

/*SMTP 测试 版 发 送 邮 件 方式 ， 使 用 SMTP 作为 服务 器 

Str = new Zend_Mail_Transport_Smtp(192.168.1.247"); 

$mail = new Zend_Mail(); 

$mail->addTo(cym3100@163.com', 获 取 用 户 注册 激活 码 '); 
$mail->setFrom('cym3100@163.com',"' 阴 日 科技 典型 模块 程序 测试 邮箱 ， 葆 喜 您 用 户 注册 成 功 ! "); 
$mail->setSubject(' 获 取 注 册 用 户 的 激活 码 '); 

$mail->setBodyHtml($mailbody); 

$mail->send($tr); 

中 

/* 本 地 网 络 版 发 送 邮 件 方法 

$config = array(auth' => 'login ， 

"username' => 'mrsoft@mrsoft.com', 


'password' => ' mingri'); /定义 SMTP 的 验证 参数 
$transport = new Zend_Mail_Transport_Smtp(localhost', $config); /实例 化 验证 的 对 象 
$mail = new Zend_Mail('GBK'); /实例 化 发 送 邮 件 对 象 
$mail->setBodyHtml($mailbody); /发 送 邮 件 主体 


$mail->setFrom($envelope, ' 明 日 科技 典型 模块 程序 测试 邮箱 ， 茶 喜 您 用 户 注册 成 功 ! ');// 定 义 邮件 发 送 使 用 的 邮箱 
$mail->addTo($_GET[email], ' 获 取 用 户 注册 激活 码 '); /定义 邮件 的 接收 邮箱 


$mail->setSubject(' 获 取 注 册 用 户 的 激活 码 '); // 定 义 邮件 主题 
$mail->send($transport); /执行 发 送 操作 
/网 络 版 发 送 邮 件 方法 


$sql = "insert into tb_member(name,password,question,answer,email,realname,birthday,telephone,qq) 
values(".trim($_GET[name]).",".md5(trim($_GET[pwd])).",".$_GET[question].",".$_GET[answen]."， 
".$_GET[email].",".$_GET[realname'].",".$_GETTbirthday].",".$_GET[telephone].",".$_GET[qq].”)”; 
$num = $conne->uidRst($sql); 

if($num == 1X{ 

$reback = "1"; 

} 

echo $reback; 

?> 


1.3.5 注册 用 户 激活 


在 用 户 注册 成 功 后 ， 需 要 登录 用 户 填 写 的 邮箱 ， 浏 览 注 册 时 发 送 的 邮件 ， 并 且 单 击 超 链接 ， 链 接 
到 activation.php 文件 , 根据 超 链接 传递 的 用 户 名 和 密码 完成 对 指定 用 户 表 中 active 字段 值 的 更 新 操作 ， 
即 完成 注册 用 户 的 激活 操作 。 

登录 mrsoft@mrsoft.com 邮箱 ， 将 查看 到 如 图 1.9 所 示 的 邮件 内 容 。 


a 
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入 webMail|powered 


文 作 D。 编 和 日 。 查看 W。 凡夫 向 工具 m 帮助 0) 
遍 W 夫 ”| 加 |- | 回 webMail-x | 励 HTTP 404 未 芥 - 园 - 己 吕 -Imp~ 交 人 7 IROv OO 
区 winmail mrsoft@amrsoftcom | 首页 | 设置 - 接 疆 ] 退出 二 
SERVER 
Ea [RE EE | 


世 写 邮件 
些 收 邮件 
局 地 址 笨 
收 件 箱 雹 时 间 2013/01/08 10:37:02 (Tue) 中 
已 发 送 收 件 人 “这 了 用 户主 且 潮 壬 " <mrsofl@msoftcom> 
EN 注册 成 功 。 你 的 娄 舌 到 是: httpJhocalhost32TWO1jactvation php? 
草稿 宁 name=mrbecd&pwd=e0cb79c341bf2e32df808be7bb46ec 人 
请 点 击 访 地 址 ， 激 舌 您 的 月 户 ! 
Hi | 了 


本 地 Intranet | 全 PA 模式: 区 用 
图 1.9 激活 邮件 的 运行 结果 


当 用 户 进入 激活 页 (activation.php〉 后 ， 激 活页 将 根据 name 和 pwd 字段 来 查找 匹配 的 注册 用 户 ， 
如 果 找 到 将 进行 更 新 操作 。 激 活 成 功 后 , 将 用 户 名 保存 到 Session 中 ,并 跳 转 到 相应 的 页 面 。 代 码 如 下 : 


石 ”> mlo% > | 
j 


<?php 

Session_start(); 

header('Content-Type:text/html;charset=gb2312"); 

include_once("conn/conn.php"); 

if (lempty($_GET[name']) && lis_null($_GET[name])X /激活 注册 用 户 
$num=$conne->getRowsNum("select * from tb_member 

where name='".$_GET[name]." and password =".$_GET[pwd]."”); 

if ($num>0){ 

$upnum=$conne->uidRst("update tb_member set active = 1 

where name='".$_GET[name]." and password = ".$_GET[pwd].""); 

if($upnum > OX 

4$4_SESSION[name']=$_GET[name]; 

echo "<script>alert(' 用 户 激活 成 功 ! ');window.location.href='main.php';</script>"; 
}else{ 

echo "<script>alert(' 您 已 经 激活 ! ');window.location.href='main.php';</script>"; 

} 

}else{ 

echo "<script>alert(' 用 户 激活 失败 ! ');window.location.href='register.php';</script>"; 
} 

} 


?> 


1.3.6” 免 激活 用 户 注册 


1.3.5 节 介 绍 的 用 户 注册 功能 中 ， 需 要 将 激活 信息 以 邮件 的 形式 发 送 到 用 户 指定 的 邮箱 中 进行 激活 
后 ， 用 户 才 可 以 使 用 。 这 里 向 读者 介绍 一 种 不 需要 激活 的 用 户 注册 方式 ， 同 样 以 上 面 注册 提交 的 数据 
为 基础 ， 只 需要 修改 用 户 注 册 提 交 数 据 的 处 理 方式 ， 就 可 以 实现 免 激 活 的 用 户 注 册 功 能 。 该 操作 在 
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register_check.php 文件 中 完成 。 关 键 代 码 如 下 : 


<?php 
include_once 'conm/conn.php'; /包含 数据 库 连 接 操作 文件 
S$reback = '0'; // 定 义 初始 变量 


$sql = "insert into tb_member(name,password,question,answer,email,realname,birthday,telephone,qq,active) 
values(".trim($_GET[name]).""".md5(trim($_GET[pwd])). _GET[question].",".$_GET[answer]."， 
".$_GET[email].",".$_GET[realname].",".$_GET[birthday].",".$_GET[telephone].",".$_GET[qq].","1")"; 


$num = $conne->uidRst($sql); /| 执行 添加 语句 
if($num == 1X{ 

$reback = "1'; 

} 

echo $reback; // 输 出 变量 值 
?> 


py 
四 在 本 模块 中 要 使 用 免 激活 模式 来 注册 用 户 ， 首 先 应 该 在 js\registerjs 文件 中 ， 修 改 注册 数 
据 处 理 URL 路 径 ， 将 注册 数据 处 理 文件 ( register_chk.php ) 修改 为 register check.php。 代 码 如 下 : 
$(regbtn').onclick = function(){ 
S$(imgdiv').style.visibility = visible'; 
url = register_chk.php?name='+$(regname').value+'&pwd='+$(regpwd1').value+'&email='+$(email).value; 
url += '&question=" +$(question').value+'&answer='+$(answer).value; 
ur += '&realname=' +$('realname’).value+'&birthday="+$('birthday’).value; 
Url += '&telephone='+$(telephone').value+'&qq='+$(qq).value; 
xmlhttp.open('get",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4){ 
if(xmlhttp.status == 200){ 
msg = xmlhttp.responseText; 
if(msg == "1'X{ 
alert(' 注 册 成 功 ， 请 到 您 的 邮箱 中 获取 激活 码 !"); 
location='index.php'; 
jelse if(msg == -1){ 
alert(' 您 的 服务 器 不 支持 Zend_mail， 或 者 邮箱 填写 错误 。 请 仔细 检查 ! ! '); 
}else{ 
alert(msg); 


} 
S$(imgdiv').style.visibility = "hidden '; 
时 


} 
xmlhttp.send(null); 
上 


如 此 ， 在 执行 用 户 注册 数据 的 处 理 时 ， 就 会 调用 上 面 重新 编写 的 register_check.php 文件 ， 用 户 注 
册 成 功 后 可 以 直接 登录 ， 而 不 必 再 进行 激活 操作 。 
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1.4 用 户 登 录 


1.4.1 用 户 登录 功能 概述 


用 户 登 录 功 能 就 是 对 用 户 输入 的 账号 、 密 码 以 及 验证 码 进行 验证 。 验 证 包括 信息 是 否 为 空 、 验 证 
码 是 否 错误 、 用 户 计算 机 是 否 已 经 被 锁定 、 用 户 是 否 存在 、 用 户 是 否 激 活 、 用 户 名 和 密码 是 否 匹 配 、 
用 户 登录 次 数 是 否 达 到 上 限 等 。 登 录 功 能 的 运行 结果 如 图 1.10 所 示 。 


了 3 主 记 编程 次 源 蝙 程 舍 验 
BE CD sanan 
: ; 
如果 您 没有 注册 ,请 您 注册 后 再 登录 。 加 果 修 已 经 注册 ,但 忘记 了 密码 ,您 可 以 通过 
找 加 密码 将 您 的 密码 找 回 。 


登录 名 称 : nr 
孙 训 同 -eeeeee 
B169 | 31 65 看 不 
验证 码 : ey 


[二 ED 医 王 本 
上 


图 1.10 单 击 “ 登 录 ” 按 钮 后 的 运行 结果 
登录 功能 包含 3 个 核心 文件 ， 分 别 是 login.php (用户 登录 页 ) 、login.js (登录 信息 验证 脚本 〉 和 
login_chk.php( 登 录 处 理 页 )。 
1.4.2 ”登录 页 面 设计 
1. 用 户 登 录 页 (login.php) 


相对 于 注册 页 面 来 说 ， 登 录 页 (login.php》 中 的 表单 元 素 要 少 很 多 ， 也 没有 那么 复杂 。 登 录 页 中 
的 元 素 名 称 和 属性 如 表 1.2 所 示 。 


表 1.2 登录 页 中 的 元 素 名 称 和 属性 


元 素 名 称 元 素 属性 值 说 了 明 
script "js/xmlhttprequest.js" 载 入 xmlhttprequest 对 象 脚本 
Script "jslogin js" 载 入 登录 信息 验证 脚本 


登录 名 称 


text 


id="lename" name="lename" 
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元 素 属性 值 
text | id="lgchk" name="lgchk" 验证 码 
img | id='chkid' sre="" width="55" height="18" 图 形 验证 码 
a | iq"changea" 更 换 验证 码 超 链接 
hidden | id="chknm" name = "chknm" value="" 隐藏 域 ， 保 存 验证 码 
button | idebm' “登录 ”按钮 
button | ideer 跳 转 到 注册 页 


id="found" 找 回 密码 


2. 登录 信息 验证 脚本 (login.js) 

(1) 在 登录 页 login.php 中 ， 载 入 了 loginjs 脚本 文件 ， 通 过 该 脚本 文件 完成 对 登录 用 户 的 所 有 验 
证 操作 。 代 码 如 下 : 

function $(idX{ 


return document.getElementByld(id); 


二 


window.onload = function(){ 


点 移动 到 最 后 一 个 文本 框 即 验证 码 输入 框 后 ， 再 次 按 下 Enter 键 ， 将 调 
用 chklg0) 函 数 对 用 户 输入 的 信息 进行 验证 。 实 现 原理 将 在 1.6 节 进 行 详细 讲解 。 代 码 如 下 : 


$(lgname')focus(); // 光 标定 位 到 用 户 名 输入 框 

$(lgname'").onkeydown = function(){ // 处 理 用 户 名 输入 框 的 键盘 按 下 事件 
if(event.keyCode == 13){ // 当 按 下 Enter 键 时 ， 光 标 自 动 下 移 
$(Ilgpwd'").select(); /密码 框 获取 焦点 

} 

$(lgpwd').onkeydown = function(){ 

if(event.keyCode == 13){ // 处 理 密码 框 的 鼠标 按 下 事件 
$('lgchk').select(); // 验 证 码 文本 框 获得 焦点 

} 

} 

$(lgchk).onkeydown = function(){ // 处 理 验证 码 输入 框 的 鼠标 按 下 事件 

if(event.keyCode == 13){ 

chklg(); // 调 用 chklg() 函 数 

} 

} 


$(lgbtn').onclick = chklg; 


(3) 进行 最 基本 的 检测 。 包 括 用 户 名 要 合法 、 信 息 不 能 为 空 、 验 证 码 输入 正确 等 。 代 码 如 下 : 


function chklg(X 
/判断 用 户 名 是 否 合法 ， 用 户 名 第 一 位 必须 是 英文 字母 ， 不 允许 有 特殊 字符 */ 


3 
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if($(lgname'").value.match(/^[a-zA-Z_]\w*$/) == nullX{ 
alert(' 请 输入 合法 名 称 '); 

$(lgname').select(); 

return false; 


/用 户 名 是 否 为 空 %/ 
if($(lgname').value == "){ 
alert(' 请 输入 用 户 名 !"); 
$(lgname').focus(); 
return false; 


/密码 是 否 为 空 % 
if($(lgpwd').value == "){ 
alert(' 请 输入 密码 !"); 
$('lgpwd').focus(); 
return false; 


} 

"验证 码 是 否 为 空 */ 
if($('lgchk').value == "){ 
alert(' 请 输入 验证 码 '); 
S$('lgchk').select(); 
return false; 


} 

人/* 验 证 码 输入 是 否 正确 */ 
if($(lgchk').value != $('chknm').value){ 
alert(' 验 证 码 输 入 错误 '); 
$('lgchk').select(); 

return false; 


} 


(4) 判断 客户 端的 Cookie 值 是 否 大 于 3， 如 果 大 于 3, 说 明 用 户 是 非法 操作 的 , 将 停止 继续 执行 。 
代码 如 下 : 


count = document.cookie.split(:")[0]; // 获 取 Cookie 信息 
if(count.split(=")[1] >= 3X 

alert(' 因 为 您 的 非法 操作 ， 您 将 无 法 再 执行 登录 操作 '); 

return false; 


} 
(5) 将 用 户 输入 的 信息 传递 给 login_chk .php 页 进行 处 理 。 代 码 如 下 : 


url = login_chk.php?act='+(Math.random())+&name='+$(lgname').value+'&pwd='+$(lgpwd').value; 
xmilhttp.open('get ,urltrue); 

xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readystate == 4){ 

if(xmlhttp.status == 200){ 

// 获 取 login_chk.php 页 的 返回 值 


@ 
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msg = xmlhttp.responseText; 
if(msg == '0){ 
alert(' 您 还 没有 激活 ， 请 先 登 录 邮箱 进行 激活 操作 。'); 
}else if(msg == "1'){ 
alert( 用 户 名 或 密码 输入 错误 ， 您 还 有 2 次 机 会 ); 
$(Igpwd ).select(); 
}else if(msg == '2){ 
alert(' 用 户 名 或 密码 输入 错误 ， 您 还 有 1 次 机 会 ); 
$('lgpwd').select(); 
} else if(msg == '4'X{ 
alert( 用 户 名 输入 错误 '); 
$(lgname').select(); 
} else if(msg == -1){ 
alert(' 登 录 成 功 '); 
location = 'main.php'; 
}else{ 
alert(msg); 
} 
; 
} 


} 
xmlhttp.send(null); 


1.4.3 ”登录 处 理 页 


在 登录 处 理 页 (login_chk.php) 中 , 对 表单 中 提交 的 用 户 名 、 密码 进 行 验证 , 包括 用 户 名 是 否 存在 、 
用 户 名 是 否 被 激活 、 用 户 登录 次 数 、 密 码 是 否 正 确 等 ， 最 终 返 回 验证 结果 。 代 码 如 下 : 


<?php 

session_start(); /开启 Session 支持 
header('Content-Type:text/html;charset=gb2312'); /| 设置 头 信息 
include_once 'conn/conn.php'; // 载 入 数据 库 类 
$name = $_GET[name]; 

$pwd = $_GET['pwd']; // 获 取 名 称 和 密码 


if(lempty($name) and lempty($pwd)}{ 

// 得 到 当前 数据 表 中 的 count 和 active 字段 

$sql = "select name,count,active from tb_member where name = ".$name.""; 
$active = $conne->getFields($sql,2); 

$count = $conne->getFields($sql,1); 

$conne->close_rst(); 

// 判 断 用 户 是 否 存在 

if($active == "X{ 

if(is_null($_COOKIE['count") or $_COOKIE[count] == OX{ 
setcookie(count ,1); 

}else{ 

setcookie(count,$_ COOKIE['count]+1); 
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} 

S$reback = 4; 

}else if($active == 0X{ // 判 断 用 户 是 否 被 激活 

$reback = '0'; 

}else if($count >= 3){ 

Sreback = '3"; 1/ 判断 用 户 的 登录 次 数 

}else{ 

$sql .= " and password = ".md5($pwd).""; 

S$num = $conne->getRowsNum($sql); 

if($num == 0 or $num == "X{ /如 果 用 户 密码 错误 ， 登 录 次 数 加 1 
S$num = $conne->uidRst("update tb_member set count = ".($count+1)." where name = ".$name.""); 
$reback = ($count+1); 

}else{ /登录 成 功 ， 清 空 count 字段 值 
if($count != OX 

$num = $conne->uidRst("update tb_member set count = 0 where name = ".$name.""); 


} 

// 设 置 Cookie 

if(isset($_COOKIE['count]) and $_COOKIE['count] {= 0){ 
setcookie('count',0); 


} 

$_SESSION['name'] = $name:; 
S$reback = "1"; 

} 

} 

} 

echo $reback; 

?> 


1.4.4 生成 及 刷新 验证 码 


验证 码 生 成 应 用 的 是 GD2 函数 库 中 的 函数 ， 操 作 存 储 于 valcode.php 文件 中 ， 其 完整 代码 可 以 参 
考 1.6 节 ， 这 里 不 再 袭 述 。 

当 页 面 第 一 次 被 载 入 时 ， 将 执行 login.js 中 的 showval() 函 数 来 生成 验证 码 ， 并 且 调 用 valcode.php 
文件 以 图 像 形式 输出 验证 码 。 因 为 验证 码 中 带 有 干扰 串 ， 如 果 用 户 看 不 清 ， 可 以 单 击 首页 中 的 “看 不 
清 ” 超 链接 来 再 次 执行 showval0 函 数 ， 即 刷新 验证 码 。login.js 脚本 中 showval0 函 数 的 代码 如 下 : 


showval(); ll 调用 showval() 函 数 生成 图 文 验 证 码 
$(changea').onclick = showval; // 更 换 验 证 码 
// 验 证 码 生成 函数 
function showval(){ 
num="; 
for(i=0;i<4;i++){ 
i /生成 随机 数 
} 
$(chkid).src='valcode.php?num='+num; /显示 图 片 验 证 码 
$(‘chknm').value = num; /将 验证 码 保存 到 chknm 隐藏 域 


@ 
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1.5 找 回 密码 


找 回 密码 功能 概述 


回 密码 功能 用 到 的 技术 也 是 邮箱 技术 。 用 户 输入 正确 的 


账号 、 密 保 问题 、 密 保 答案 后 ， 系 统 将 随机 生成 的 一 组 数字 发 
送 到 用 户 的 邮箱 中 ， 用 户 可 以 通过 邮箱 来 得 到 新 密码 。 找 回 密 
码 页 面 的 运行 结果 如 图 1.11 所 示 。 


1.5.2 


入 


弹出 的 


找 回 密码 功能 实现 过 程 


回 密码 操作 从 单 击 主页 中 的 “ 找 回 密码 ”按钮 开始 ， 在 
found.php 文件 中 创建 一 个 表单 ， 提 交 要 找 回 的 账号 


密 保 问题 和 密 保 答 案 ,， 并 且 载 入 xmlhttprequestjs 和 foundjs 脚 


本 文件 。 


found.php 的 关键 代码 如 下 : 


和 eee Wnt [eve] 


找 回 账号 x 


密 保 问题 : [ 国 日 科技 
密 保 答案 : 图书 开发 


图 Imtemnet | 保护 模式 茜 级” 页 100% > 
TT J 


图 1.11 找 回 


密码 页 面 的 运行 结果 


<link rel="stylesheet" href="css/style.css" /> 

<script language="javascript" src="js/found.js"></script> 

<script language="javascript" src="js/xmlhttprequest.js"></script> 
<div id="fdbgdiv" > 

<div id="top">&nbsp;>> 密 码 找 回 </div> 

<div id="foundnamediv"> 找 回 账号 : 

<input id="foundname" type="text" style=" width: 100px; height:15px; border:1px #000000 solid;" /></div> 
<div id="foundnamediv"> 密 保 问 题 : 
<input id="fdquestion" type="text" style=" width: 100px; height:15px; border1px #000000 solid;" /></div> 
<div id="foundnamediv"> 密 保 答案 : 
<input id="fdanswer" type="text" style=" width: 100px; height:15px; border: 1px #000000 solid;" /></div> 
<div id="foundnamediv" align="center"><button id="step1"></button></div> 
</div> 


在 found.js 文件 中 ， 调 用 found_chk.php 文件 ， 完 成 密码 的 找 回 操作 。 关 键 代码 如 下 : 


function $(idX{ 
return document.getElementByld(id); 


window.onload = function(){ 
$(foundname’).focus(); 
$('step1').onclick = function(){ 
if($(foundname').value != " && $('fdquestion').value != " && $(fdanswer).value != "X{ 
xmlhttp.open(get',found_chk.php?foundname='+$(foundname'). 
value+'&question='+$(fdquestion').value+'&answer='+$(fdanswer).value,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4 && xmlhttp.status == 200X{ 
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msg = xmlhttp.responseText; 
if(msg == 1){ 

alert(' 找 回 密码 成 功 ， 请 登录 邮箱 取 回 密码 !"); 
window.close(); 

}else{ 

alert(' 填 写 信息 错误 ! '); 

} 

} 

} 

xmilhttp.send(null); 

}else{ 
alert(' 请 填写 完整 信息 '); 
$(foundname').focus(); 
return false; 

下 

} 

} 


在 found_chk.php 文件 中 , 对 用 户 提交 的 密 保 问 题 和 密 保 答案 进行 验证 , 如 果 正 确 则 生成 新 的 密码 ， 
并 且 更 新 数据 库 中 存储 的 密码 ， 同 时 将 新 密码 发 送 到 用 户 注册 时 填写 的 邮箱 中 。found_chk.php 的 关键 
代码 如 下 : 


<?php 

include_once 'conm/conn.php'; 

require_once 'Zend/Mail.php'; // 调 用 发 送 邮件 的 文件 
require_once 'Zend/Mail/Transport/Smtp.php'; // 调 用 SMTP 验证 文件 
$reback = '0'; 

$name = $_GET[foundname']; /获取 用 户 名 
$question = $_GET[question'; /获取 密 码 保护 问题 
$answer = $_GET[answer]; // 获 取 密 码 保护 的 答案 


$sql = "select email from tb_member 
where name = ".$name." and question = ".$question." and answer = ".$answer.""; // 定 义 查询 语句 


$email = $conne->getFields($sql,0); 1/ 判断 用 户 提交 的 密 保 问题 和 答案 是 否 正确 
if($email != "X // 如 果 正 确 则 执行 如 下 操作 
$md = rand(1000,time()); /| 生成 随机 数 


$sql = "update tb_member set password = ".md5($rnd)." 

where name = ".$name." and question = ".$question." and answer = ".$answer.""; // 更 新 密码 
$tmpnum = $conne->uidRst($sql); /执行 更 新 操作 

if($tmpnum >= 1){ 

/发 送 密码 邮件 

$subject=" 找 回 密 码 "; 

$mailbody=' 密 码 找 回 成 功 。 您 账号 的 新 密码 是 .$md; 
/l/$envelope["from"]="cym3100@163.com"; 

$envelope="mrsoft@mrsoft.com"; // 网 络 版 定义 登录 使 用 的 邮箱 
ASMTP 测试 版 发 送 邮件 方式 ， 使 用 SMTP 作为 服务 器 */ 

$tr = new Zend_MailL Transport_ Smtp(192.168.1.247"); 

$mail = new Zend_Mail(); 

$mail->addTo($email,' 获 取 用 户 新 密码 '); 

$mail->setFrom($envelope, 阴 日 科技 典型 模块 程序 测试 邮箱 ， 修 改 用 户 注册 密码 ! "); 


@ 
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$mail->setSubject($subject); 
$mail->setBodyHtml($mailbody); 
li$mail->send($tr); 

让 网 络 版 发 送 邮 件 方法 */ 

"$config = array(auth => 'login', 
"username' => 'mrsoft@mrsoft.com', 


'password' => ' mingri'); // 定 义 SMTP 的 验证 参数 

S$transport = new Zend_Mail Transport_Smtp(smtp.sohu.com', $config); // 实 例 化 验证 的 对 象 
$mail = new Zend_Mail('GBK'); /实例 化 发 送 邮 件 对 象 
$mail->setBodyHtml($mailbody); // 发 送 邮 件 主 体 


$mail->setFrom($envelope, ' 阴 日 科技 典型 模块 程序 测试 邮箱 ， 修 改 用 户 注册 密码 !'); 
$mail->addTo($email, 获取 用 户 新 密码 '); /定义 邮件 的 接收 邮箱 
$mail->setSubject($subject); /定义 邮件 主题 
$mail->send($transport); /|/ 执 行 发 送 操作 */ 

让 网 络 版 发 送 邮 件 方法 */ 

if(false ==$mail->send($tr) X{ 

$reback = -1; 

}else{ 

$reback = "1'; 

} 

}else{ 

$reback = '2', 

} 

}else{ 

$reback = $sql; 

} 

echo $reback; 

?> 


1.6 技术 提炼 


1.6.1 防 SQL 注入 技术 


SQL 注入 ， 是 指 一 些 精通 SQL 语句 的 用 户 ， 通 过 在 表单 或 浏览 器 地 址 栏 中 输入 SQL 语句 来 绕 过 
系统 检验 的 一 种 技术 。 一 般 防 范 SQL 注入 的 手段 是 过 滤 敏 感 字 符 ， 例 如 引号 等 。 

PHP 相对 于 ASP 来 说 ， 要 安全 得 多 ， 但 这 不 代表 PHP 网 站 就 不 会 出 现 SQL 注入 的 情况 。 幸 运 的 

是 , 在 PHP 中 防 SQL 注入 ， 要 比 ASP 简单 、 方 便 ， 不 需要 写 一 大 段 的 转换 语句 ， 只 要 几 个 函数 即 可 。 

(1) intval0 函 数 

intval0) 函 数 的 作用 是 返回 变量 的 整数 值 。 其 语法 如 下 : 


int intval( mixed var [, int base] ) 


(2) addslashes() 函 数 
addslashes(0) 函 数 就 是 在 操作 数据 库 时 ， 对 其 中 的 特殊 字符 进行 自动 转 义 ， 即 在 特殊 字符 前 加 上 反 


@ 
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斜 杠 (\) ， 包 括 单 引号 () 、 双 引号 (") 、NULL， 但 是 不 包括 “%” 和 “_”。 该 函数 的 语法 如 下 : 
string addslashes( string str ) 


此 外 ， 还 可 以 使 用 mysql_real_escape_string0) 来 进行 转 义 ， 效 果 和 addslashes() 函 数 是 一 样 的 。 
1.6.2 Ajax 无 刷新 验证 技术 


Ajax (Asynchronous JavaScriptand XML， 异 步 JavaScript 和 XML ) 是 时 下 最 流行 的 技术 。Ajax 不 
是 新 的 技术 ， 而 是 原 有 技术 的 集合 ， 这 从 它 的 名 字 上 就 能 够 看 出 来 。 

Ajax 的 核心 技术 是 xmlHttpRequest。 通过 xmlHttpRequest 中 的 open0 和 send() 方 法 ， 可 以 在 不 刷新 
当前 页 面 的 情况 下 向 处 理 页 发 送 数据 ; 通过 xmlHttpRequest 中 的 responseText 和 responseXML 属性 ， 
可 以 得 到 处 理 页 的 输出 结果 。 

Ajax 之 所 以 能 够 流行 ， 是 因为 它 能 为 用 户 提供 良好 的 交互 性 ， 这 个 特性 在 本 模块 中 被 充分 地 体现 
出 来 。 注 册 用 户 无 须 经 过 “漫长 ”的 等 待 就 可 以 知道 输入 的 账号 是 否 可 用 ， 登 录用 户 也 不 必 担 心 因 为 
输 错 登录 信息 而 重新 刷新 整个 页 面 。 

使 用 Ajax， 一般 分 为 下 面 几 步 。 

(1) 创建 xmlHttpRequest 对 象 。 不 同 的 浏览 器 ， 创 建 xmlHttpRequest 对 象 及 使 用 的 方法 有 一 些 差 


别 ， 这 里 只 针对 IE 浏览 器 进行 创建 。 代 码 如 下 : 
var xmlhttp = false; 1/ 初始 化 变量 
// 如 果 ActiveXObject 存在 ， 说 明 是 IE 1.0 以 上 的 版 本 ， 否 则 使 用 XMLHttpRequest 创建 
if(window.ActiveXObject}{ 


xmlhttp = new ActiveXObject("Microsoft.XMLHTTP"); 
}else if(window.XMLHttpReuqest}{ 

xmlhttp = new XMLHttpRequest(); 

} 


(2) 对 和 象 创建 成 功 后 ， 就 可 以 使 用 对 象 中 的 open0 方 法 创建 新 请 求 了 。 该 方法 的 格式 如 下 : 
xmlhttp.open(rmethod, rurl,isAsync); 


rmethod 参数 指明 请 求 的 方法 ， 如 get 或 post 
rurl 参数 指明 请 求 的 页 面 。 可 以 是 绝对 地 址 ， 也 可 以 是 相对 地 址 。 
isAsync 参数 指明 请 求 是 否 为 异步 。 默 认为 tue， 即 异步 。 
(3) 如 果 isAsync 等 于 true， 那 么 当 请 求 的 状态 改变 时 ， 将 调用 onreadystatechange 属性 ， 该 属性 
指定 了 一 个 回调 函数 。 格 式 如 下 : 


xmlhttp.onreadystatechange = reabackfunc; 


或 者 
xmlhttp.onreadystatechange = function(){…} 


(4) 在 回调 函数 中 ， 首 先 需要 判断 http 的 请 求 状态 和 http 状态 码 。 这 是 通过 readystate 和 statues 
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属性 来 判断 的 。readystate 属性 有 5 种 状态 值 ， 常 用 的 是 4， 表 示 数 据 接收 完毕 。statues 属性 的 值 比较 
多 ， 常 用 的 是 200， 表 示 请 求 成 功 。 一 般 使 用 这 两 个 属性 一 起 来 判断 。 格 式 如 下 : 


xmlhttp.onreadystatechange = function(){ 
if(readystate == 4 and statues == 200X{ 


} 
} 


(5) 当 响 应 页 处 理 结束 后 ， 即 满足 了 “readystate==4 and statues == 200” 这 个 条 件 ， 就 可 以 使 用 
xmlHttpRequest 对 象 的 属性 获取 响应 页 的 值 了 。 常 用 的 有 responseText、responseXML、responseStream 
等 ， 这 里 以 responseText 为 例 进行 介绍 。 

responseText 属性 是 将 响应 页 的 输出 信息 作为 字符 串 返回 。 格 式 如 下 : 


str = xmlhttp.responseText; 

(6) 使 用 send() 方 法 来 接收 回应 。send( 方 法 可 以 传递 数据 ， 但 这 取决 于 open() 方 法 中 的 rmethod 
参数 ， 当 参数 为 get 时 ， 数 据 是 附 在 URL 中 进行 传递 的 ， 当 参数 为 post 时 ， 数 据 只 能 使 用 send() 方 法 
进行 传递 。send() 方 法 的 语法 如 下 : 

xmlhttp.send([rdate]); 
1.6.3 ”验证 码 技术 

验证 码 技术 ， 是 为 了 防止 用 户 名 和 密码 被 暴力 破解 ， 在 登录 页 面 上 生成 一 组 随机 数 ， 每 次 刷新 页 
面 时 ， 随 机 数 都 会 改变 。 一 般 是 4 位 ， 也 有 更 多 位 的 。 

1. 获取 验证 码 


PHP 中 的 验证 码 可 以 通过 rand() 函 数 生成 随机 数 的 方式 得 到 。rand() 函 数 可 以 获取 指定 范围 内 的 随 
机 数 。 该 函数 的 语法 如 下 : 


int rand([int min, int max]) 


如 果 省 略 两 个 参数 ， 那 么 将 返回 0 到 RAND_MAX 之 间 的 随机 整数 ， 否 则 ， 返 回 min 和 max 之 间 
的 整数 。 例 如 本 例 中 要 获取 4 位 十 六 进 制 的 整数 ， 代 码 如 下 : 

<?php 

for($i=0;$i<4;$i++X{ 

$num .= dechex(rand(0,15)); /| 生成 随机 数 

让 


?> 


函数 dechex() 可 以 将 参数 转换 为 十 六 进 制 表 示 。 
使 用 JavaScript 也 可 以 生成 十 六 进 制 随机 数 ， 但 是 稍 有 些 复 杂 。JavaScript 中 不 能 直接 将 十 进 制 数 
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转 为 十 六 进 制 ， 需 要 手动 进行 转换 。 首 先 使 用 Math.random0 函 数 生成 0 一 15 之 间 的 随机 数 ， 然 后 使 用 
Math.ceil0 函 数 将 随机 数 取 整 ， 接 下 来 就 要 逐次 判断 该 值 ， 如 果 该 值 大 于 9,， 那么 将 10 一 15 的 数 一 一 对 
应 转换 为 a、b、c、d、e、f。 转换 完成 后 ， 将 值 累加 ， 最 后 传 给 valcode.php 页 。 

使 用 JavaScript 生成 十 六 进 制 随机 数 的 完整 代码 如 下 : 


// 生 成 随机 数 
function showval(){ 
num="; 
for(i=0;i<4;i++){ 
tmp= Math.ceil((Math.random()* 15)); 
if(tmp > 9X 
switch(tmpX{ 
case(10): 

num += 'a'; 
break; 
case(11): 

num += 'b'; 
break; 
case(12): 

num += "cy 
break; 
case(13): 

num += 'd"; 
break; 
case(14): 

num += 'e' 
break; 
case(15): 

num += TF; 
break; 

上 

}else{ 

num += tmp; 


$(chkid').src='valcode.php?num='+num; 
$(‘chknm').value = num; 


, 


/循环 输出 4 位 验证 码 
// 取 得 一 位 十 六 进 制 的 整数 
/依次 判断 随机 数 


// 如 果 随机 数 等 于 10， 换 为 a 


// 如 果 随机 数 等 于 11， 换 为 b 


// 如 果 随机 数 等 于 12， 换 为 c 
// 如 果 随机 数 等 于 13， 换 为 d 


// 如 果 随 机 数 等 于 14， 换 为 e 


// 如 果 随机 数 等 于 15， 换 为 f 


// 将 生成 的 随机 数 传 给 图 像 生成 页 
// 将 随机 数 的 值 保 存 到 页 面 的 隐藏 域 中 


2. 显示 随机 数 图 片 


显示 随机 数 的 方式 很 多 ， 将 随机 数 写 入 一 个 图 片 中 再 显示 是 目前 常用 的 方法 。 在 PHP 中， 可 以 使 
用 GD 函数 库 来 实现 。 使 用 到 的 函数 主要 有 imagecreate() 、imagecolorallocate()、imagestring( 、 
imagesetpixel0、imagepng(0 和 imagedestroy0 函 数 。 


(1) imagecreate() 函 数 


imagecreate() 函 数 用 来 创建 一 个 基于 调 色 板 的 空白 图 像 源 ， 这 是 生成 图 片 的 第 一 步 。 该 函数 的 语法 


如 下 : 


@ 
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resource imagecreate( int width, int height ) 


参数 width 和 height 分 别 指定 了 图 像 的 宽 和 高 。 
(2) imagecolorallocateO) 函 数 
imagecolorallocateO 函 数 可 以 为 创建 后 的 图 像 分 配 颜 色 。 该 函数 的 语法 如 下 : 


int imagecolorallocate( resource image, int red, int green, int blue ) 


参数 image 是 一 个 图 像 源 。 

参数 red、green 和 blue 表示 红 、 黄 、 蓝 三 元 素 的 成 分 。 每 种 颜色 的 取 值 范围 在 1 一 255 之 间 。 
(3) imagestring() 函 数 

图 像 创建 完成 后 ， 就 可 以 使 用 imagestring() 函 数 来 添加 图 像 文字 了 。 该 函数 的 语法 如 下 : 


bool imagestring( resource image, int font, int x, int y, string s, int col ) 


参数 image 是 一 个 图 像 源 。 
参数 font 可 以 设置 字体 ， 如 果 使 用 系统 默认 字体 ， 可 以 使 用 1 一 5 的 数字 。 
参数 x 和 yy 分别 表示 文字 相对 于 整 幅 图 像 的 x 轴 和 Yy 轴 坐标 , 即 所 输入 的 字符 串 的 左上 角 坐 标 。 
参数 s 是 要 显示 的 字符 串 。 
参数 col 为 字体 颜色 ， 也 是 使 用 imagecolorallocate(0 函 数 来 分 配 。 
(4) imagesetpixel0 函 数 
使 用 imagecolorallocate() 函 数 创 建 的 是 一 个 单一 背景 色 的 图 像 ， 如 果 希 望 向 图 像 中 添加 干扰 码 ， 可 


办 办 办 办 轨 


以 使 用 imagesetpixel0 函 数 ， 该 函数 的 作用 是 画 一 个 像素 点 。imagesetpixel() 函 数 的 语法 如 下 : 


bool imagesetpixel( resource image, int x, int y, int color ) 
参数 说 明和 imagestring() 函 数 相似 ， 这 里 不 再 更 述 。 
(5) imagepng() 函 数 
imagepng() 函 数 将 创建 完成 的 图 片 以 .png 的 格式 输出 。 该 函数 的 语法 如 下 : 


bool imagepng( resource image [, string filename] ) 


参数 image 是 要 保存 的 图 像 源 。 

参数 filename 是 要 保存 的 图 像 名 。 如 果 省 略 ， 则 直接 输出 到 浏览 器 。 
(6) imagedestroy() 函 数 

图 像 保存 完毕 后 ， 使 用 imagedestroy0 函 数 来 释放 内 存 。 该 函数 的 语法 如 下 : 


bool imagedestroy( resource image ) 
验证 码 的 生成 文件 是 valcode.php。 其 完整 代码 如 下 : 
<?php 


header("content-type:image/png"); // 设 置 页 面 编码 
$num =$_GET[num]; /获取 超 级 链接 传递 的 随机 数 
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<? gewidth=60; // 定 义 画布 的 宽 
S$imageheight=18; /定义 画布 的 高 
$numimage = imagecreate($imagewidth,$imageheight); /创建 画布 
imagecolorallocate($numimage,240,240,240); // 设 置 画布 颜色 
for($i=0;$i<strlen($num);$i++){f /| 循环 读 取 随机 数 


$x = mt_rand(1,8)+$imagewidth*$i/4; 
$y = mt_rand(1,$imageheight/4); 
$color=imagecolorallocate($numimage,mt_rand(0,150),mt_rand(0,150),mt_rand(0,150)); /定义 图 像 的 颜色 


imagestring($numimage,5,$x,$y,$num[$il,$color); // 将 随机 数 写 入 到 画布 中 

} 

for($i=0;$i<200;$i++j{ /for 循环 语句 生成 干扰 线 
$randcolor=imagecolorallocate($numimage,rand(200,255),rand(200,255),rand(200,255)); /定义 颜色 
imagesetpixel($numimage,rand()%70,rand()%20,$randcolor); /生成 干扰 线 

} 

imagepng($numimage); /生成 图 像 
imagedestroy($numimage); /释放 资源 

?> 


1.6.4 ”E-mail 激活 技术 


为 了 防止 恶意 注册 ， 现 在 有 很 多 网 站 采用 了 E-mail 激活 技术 。 当 用 户 注册 成 功 后 ， 并 不 是 马上 可 
以 使 用 ， 而 是 需要 先 登 录 邮 箱 ， 通 过 系统 发 出 的 E-mail 中 的 链接 进行 激活 ， 只 有 激活 后 ， 账 号 才 可 以 
使 用 。 对 于 找 回 密码 的 用 户 ， 系 统 会 将 新 密码 发 送 到 邮箱 中 ， 这 在 一 定 程度 上 提高 了 安全 性 。 

本 模块 使 用 ZendFramework 中 的 Zend_Mail 组 件 完 成 邮件 的 发 送 操作 ， 在 本 书 的 光盘 中 没有 提供 
Zend_Mail 组 件 ， 如 果 要 运行 这 个 程序 ， 需 要 读者 自己 下 载 Zend 组 件 ， 并 将 其 复制 到 本 模块 的 根 目录 
下 ， 即 mr/01 文件 夹 下 。 


& / 
% 明 在 执行 邮件 的 发 送 时 ， 笔 者 使 用 的 是 Winmail 软件 ， 邮 箱 名 称 是 mrsoft@mrsoftcom， 密 
码 是 mingri。 读 者 在 完成 用 户 注 册 之 后 ， 可 以 登录 这 个 邮箱 完成 注册 用 户 的 激活 操作 ， 否 则 注册 的 
用 户 是 不 能 够 使 用 的 。 


注册 用 户 激活 的 原理 是 控制 数据 库 中 用 户 信息 表 (tb_member) 中 active 字段 的 值 ， 注 册 成 功 后 该 
字段 的 默认 值 是 0， 但 此 时 该 用 户 是 不 能 够 登录 网 站 的 ， 必 须 通过 邮箱 激活 ， 将 该 字段 的 值 更 新 为 1 
之 后 才 可 以 使 用 。 激 活 操作 的 文件 存储 于 activation.php 文件 中 。 关 键 代码 如 下 : 


<?php 
session_start(); 
header('Content-Type:text/html;charset=gb2312'); 
include_once("conn/conn.php"); 
if (lempty($_GET[name']) && lis_null($_GET[name'"])X{ 1/ 激活 注册 用 户 
$num=$conne->getRowsNum 
("select * from tb_member where name=".$_GET['name']." and password =".$_GET[pwd'].""); 
if ($num>0X{ 
$upnum=$conne->uidRst 
("update tb_member set active = 1 where name=".$_GET[name']." and password= ".$_GET[pwd].""); 


@ 
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if($upnum > OX 
$_SESSION[name] =$_GET[name']; 
echo "<script>alert(' 用 户 激活 成 功 ! ');window.location.href='main.php';</script>"; 


}else{ 
echo "<script>alert(' 您 已 经 激活 ! ');window.location.href='main.php';</script>"; 


jelsef 
echo "<script>alert(' 用 户 激活 失败 ! ');window.location.href='register.php';</script>"; 
} 


} 
> 


1.6.5 ”应 用 键盘 响应 事件 验证 信息 是 否 合法 


新 用 户 注 册 时 ， 随 着 信息 输入 ， 系 统 实时 显示 信息 的 正确 性 ; 用 户 登 录 时 ， 无 须 使 用 鼠标 来 选择 
文本 框 ， 只 要 按 下 Enter 键 ， 就 自动 下 移 。 这 些 功能 都 是 通过 JavaScript 脚本 中 的 键盘 响应 事件 来 
实现 的 。 随 着 Ajax 技术 的 流行 ，JavaScript 已 经 成 为 一 个 开发 人 员 必 须要 掌握 的 技术 。 下 面 就 来 简单 
了 解 一 下 键盘 事件 。 

用 户 通过 onkeydown 和 onkeyup 事件 来 触发 响应 事件 。 使 用 方法 和 onclick 事件 类 似 。onkeydown 
表示 当 键 盘 上 的 键 被 按 下 时 触发 ,onkeyup 则 正好 相反 ， 当 键盘 上 的 键 被 按 下 又 抬 起 时 触发 。 在 页 面 中 
加 载 事件 的 方式 有 多 种 ， 这 里 介绍 两 种 最 常用 的 方式 。 

(1) 将 事件 直接 添加 到 页 面 元 素 中 。 这 种 方法 最 直接 、 简 单 。 格 式 如 下 : 


<script type="text/javascript"> 
function refer(){ 


} 

</script> 

<input type="text" onkeydown="refer()" /> 

当 该 用 户 输入 完 信息 后 ， 按 任意 键 ，onkeydown 事件 被 触发 ， 并 调用 refer0 函 数 。 

(2) 通过 window.onload 加 载 。 当 页 面 被 载 入 时 ， 事 件 被 载 入 。 格 式 如 下 : 
<script> 
window.onload = function(X{ 
document.getElementByld(lgname').onkeydown = function(){ 


</script> 


<input id="lgname" type="text" /> 


当 用 户 输入 信息 时 ， 每 输入 一 个 字母 ， 都 将 触发 该 事件 ， 在 该 事件 调用 的 函数 中 ， 对 用 户 输入 信 
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息 进 行 判断 , 例如 用 户 名 必须 大 于 等 于 2、 密码 最 短 6 位 、E-mail 必须 合法 等 。 验证 用 户 名 的 代码 如 下 : 


// 验 证 用 户 名 

// 为 id 等 于 regname 的 页 面 元 素 添加 onkeyup 事件 
/用 户 每 按 一 次 键 ， 都 会 调用 一 次 该 函数 
$(regname').onkeyup = function (X 


name = $(regname').value; /获取 输入 内 容 

cname2 ="; 

ifname.match(^[a-zA-Z_]7) == "多 // 判 断 输入 字符 是 否 在 有 效 范围 之 内 
$(namediv).innerHTML = '<font color=red> 必 须 以 字母 或 下 划 线 开头 </font>"; 

cname1 ="; 

}else if(name.length < 2){ /判断 输 入 字符 的 长 度 
$(‘namediv').innerHTML = '<font color=red> 注 册 名 称 必须 大 于 等 于 2 位 </font>'; 

cname1 ="; 

}else{ 


$(‘namediv').innerHTML = '<font color=green> 注 册 名 称 符合 标准 </font>"; 
cname1 ='yes'; 


有 
chkreg(); 
} 


该 段 代码 的 运行 结果 如 图 1.12 所 示 。 


注册 名 称 : | 
初始 状态 
四 注册 名 称 必须 大 于 等 于 2 位 上 @ 必须 以 字母 或 下 烛 战 开头 
按 下 第 一 个 另 一 种 效果 
mr | 注册 名 称 符合 标准 
符合 要 求 


图 1.12 验证 信息 合法 性 


使 用 onkeydown 事件 还 可 以 实现 对 特定 键 的 控制 ， 包 括 Enter 键 、Ctrl 键 、Alt 键 等 在 内 的 所 有 按 
键 ， 这 是 通过 在 onkeydown 事件 中 使 用 keyCode 属性 来 实现 的 。keyCode 属性 能 够 知道 用 户 按 下 的 是 
哪个 键 ， 例 如 Enter 键 等 于 13、 空 格 键 等 于 32 等 。 使 用 keyCode 属性 的 一 般 格式 如 下 : 

<Script> 


window.onload = function(X 
document.getElementByld(lgname') = function()f 


if(event.keyCode == 13){ 1/ 判断 用 户 是 否 按 下 了 Enter 键 
} 
y 
} 
</script> 
在 本 模块 中 , 实现 了 焦点 自动 下 移 功能 。 当 用 户 按 下 Enter 键 时 ， 页 面 的 焦点 自动 移 到 下 一 个 文本 


框 中 。 实 现 后 的 效果 如 图 1.13 所 示 。 
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小 


图 1.13 焦点 自动 下 移 的 运行 效果 
1.6.6 PHP 中 操作 Cookie 技术 


Cookie 的 作用 是 当 用 户 第 一 次 访问 某 服务 器 时 ， 服 务 器 将 一 些 信息 保存 到 客户 端 计算 机 中 。 在 以 
后 的 一 段 时 间 内 ， 当 用 户 再 次 访问 这 个 服务 器 时 ， 服 务 器 通过 Cookie 信息 ， 能 够 识别 该 用 户 。 在 很 长 
一 段 时 间 内 ，Cookie 都 被 当 作 一 个 不 安全 的 因素 ， 很 少 再 被 使 用 ， 转 而 使 用 Session。 但 是 ，Cookie 的 
功能 确实 十 分 方便 ， 如 自动 登录 、 网 站 统计 等 ， 都 要 比 其 他 的 实现 方式 要 简单 、 快 捷 得 多 。Cookie 只 
是 一 个 文本 文件 ， 不 能 访问 本 地 硬盘 ， 无 法 传播 病毒 木马 程序 。 唯 一 要 注意 的 是 ，Cookie 只 能 识别 计 
算 机 ， 而 不 会 在 意 谁 在 使 用 。 

在 PHP 中 操作 Cookie 使 用 setcookie() 函 数 和 $_COOKIE 预定 义 变量 。setcookie() 函 数 的 语法 格式 
如 下 : 


bool setcookie( string name [, string value [, int expire [, string path [, string domain [, bool secure]]]]] ) 


参数 name 设置 了 Cookie 的 名 字 。 

参数 value 是 Cookie 的 值 ， 即 参数 name 的 值 。 

参数 expire 用 来 设置 Cookie 的 过 期 时 间 。 该 参数 以 时 间 戳 的 形式 存在 。 一 般 设置 Cookie 过 期 

时 间 时 ， 通 过 “time(0+ 秒 数 ”来 实现 。 例 如 “time(+60x10”， 表 示 Cookie 将 在 10 分 钟 后 失 

效 。 设 置 path 设 马上 失效 ， 可 以 将 时 间 设 为 当前 日 期 之 前 ， 如 “time0-1”， 那 么 Cookie 会 

立即 失效 。 

参数 path 表示 Cookie 在 服务 器 端的 有 效 范围 。 如 果 path 设 为 “/”， 那 么 Cookie 在 整个 服务 
器 内 都 有 效 ， 如 果 path 设 为 “/05/”， 那 么 Cookie 只 在 服务 器 下 的 05 目录 有 效 。 

参数 domain 指定 了 Cookie 有 效 的 域名 范围 。 以 www.mrbccd.com 为 例 ， 如 果 domain 设 为 
“.mrbccd.com ”那么 Cookie 在 该 域 的 所 有 子 域 都 有 效 ; 如 果 domain 设 为 "www.mrbccd.com ”， 
那么 Cookie 只 在 www.mrbccd.com 域内 有 效 。 

参数 secure 表明 是 否 在 https 中 传送 。 如 果 是 true, Cookie 仅 在 https 连接 中 被 设置 ,默认 是 false。 


1.6.7 在 JavaScript 中 操作 Cookie 技术 


在 JavaScript 中 可 以 通过 document 对 象 中 的 Cookie 属性 对 Cookie 进行 创建 、 读 取 、 删 除 等 操作 。 
先 来 看 一 下 创建 Cookie 的 格式 〈 代 码 中 加 粗 的 部 分 是 需要 用 户 设置 的 ) 。 


办 办 轨 


documen.cookie = "Name=Value; expires=time; path=URL; domain= Domain" 


® 
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其 中 各 个 参数 的 含义 和 PHP 中 的 基本 相同 ， 这 里 不 再 歼 述 。 


在 JavaScript 中 读 取 Cookie 有 一 点 复杂 。 在 JavaScript 看 来 ，Cookie 就 是 一 个 字符 串 。 例 如 一 个 


名 字 为 count、 值 为 10 的 Cookie， 在 JavaScript 中 的 格式 是 : 
count=10 


如 果 有 多 个 Cookie， 那 么 它 的 格式 如 下 : 


count=10; name=mr; … 


而 真正 需要 的 值 ， 其 实 只 有 10 而 已 。 所 以 ， 要 获取 Cookie 的 值 ， 就 要 使 用 split0 函 数 将 Cookie 


进行 拆 分 。 获 取 count 值 的 代码 如 下 : 


<script> 


if(document.cookie=="){ /Cookie 等 于 空 时 


document.write('no cookie'); 
}else{ 


cookiearr = document.cookie.split('; "); /以 分 号 为 分 隔 符 ， 对 Cookie 进行 分 割 


leng = cookiearr.length; /获取 数组 长 度 
for(i=0; i<leng; i++X{ 


if(cookiearr[i].split('=")[0] == 'count'}{ // 判 断 哪 一 个 Cookie 等 于 count 


document.write(cookiearrli].split(=")[1]); // 输 出 count 的 值 
} 


</script> 


1.6.8 用 户 自动 登录 技术 


自动 登录 的 原理 是 : 当 用 户 打 开 登 录 页 面 时 ， 登 录 页 面 首先 判断 用 户 客户 端 机 器 中 的 Cookie 值 ， 


如 果 该 值 不 存在 ， 或 者 Cookie 已 失效 ， 将 显示 登录 表单 ; 如果 Cookie 值 存在 ， 
所 保存 的 用 户 名 进行 登录 。 


则 直接 使 用 Cookie 中 


在 本 模块 中 ， 用 户 首先 打开 的 是 index.php 页 ， 该 页 判断 Cookie 值 ， 如 果 没 有 ， 则 跳 转 到 login.php 


进行 登录 ; 如 果 Cookie 有 值 ， 则 将 Cookie 中 的 用 户 名 保存 到 Session， 直 接 进入 3 


EE 页面。 实现 代码 如 下 : 


<?php 

session_start(); /开启 Session 支持 
header('Content-Type:text/html;charset=gb2312'); /| 设置 页 面 编码 
if(lempty($_COOKIE[name']) and !is_nulll$_COOKIE[name])) /| 判断 Cookie 是 否 为 空 
$_SESSION[name'] = $_COOKIE[name1]; /将 Cookie 保存 到 Session 中 
header('location:http://localhost/model/05/01/main.php’); // 跳 转 到 main.php 页 

}else{ /Cookie 为 空 ， 说 明 没有 登录 
header(location:http:Wlocalhost:82/TM/01/ogin.php'); // 跳 转 到 login.php 页 


@ 
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这 里 8$_ COOKIE[mame"] 的 值 是 在 登录 成 功 时 设置 的 。 相 关 代码 如 下 : 


<?php 

go // 如 果 登 录 成 功 
setcookie('name',$name,time()+60*10); /| 保存 $name 的 同时 设置 时 间 
i 


1.7 本 章 小 结 


本 章 主要 讲解 的 是 注册 登录 模块 设计 过 程 ， 即 从 数据 库 设 计 到 具体 模块 的 创建 。 其 中 在 具体 模块 
创建 的 过 程 中 针对 用 户 登 录 模 块 、 用 户 注册 模块 进行 重点 讲解 ， 并 且 在 其 中 应 用 了 很 多 新 的 技术 和 技 
巧 , 例如 通过 Ajax 无 刷新 验证 技术 、 通过 在 PHP 中 操作 Cookie 技术 、 通 过 在 JavaScript 中 操作 Cookie 
技术 ， 以 及 通过 Zend_mail 函数 实现 PHP 操作 技术 等 。 


人 后 = 
性 
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网 页 计数 器 的 形式 多 种 多 样 ， 有 直接 以 数字 计数 存储 于 文本 或 数 
据 库 中 的 ， 有 以 图 形 方 式 计数 存储 于 文档 或 数据 库 中 的 。 在 浏览 网 页 
时 经 常 可 以 看 到 网 站 的 首页 带 有 一 个 制作 精美 的 图 形 计 数 器 ， 用 来 显 
示 该 网 站 的 访问 量 。 应 用 图 形 做 计数 器 ， 不 仅 可 以 美化 网 站 的 页 面 ， 
而 且 还 能 够 吸引 更 多 浏览 者 的 注意 力 ， 提 高 网 站 的 访问 量 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

WI 简单 数字 计数 路 

由 Session 实现 无 刷新 计数 路 
GD2 图 形 计数 路 
数据 库 数字 计数 路 
Cookie 计数 中 
图 形 数 字 计 数 路 


豆 吾 吾 至 
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2.1 网 页 计数 器 模块 概述 


网 站 的 计数 器 对 于 网 站 管理 者 来 说 是 一 个 非常 值得 关注 的 部 分 ， 它 记录 了 网 站 被 访问 的 次 数 ， 客 
观 地 反映 了 网 站 受 欢迎 的 程度 。 如 果 网 站 的 点 击 率 很 高 ， 访 问 者 也 会 认为 网 站 值得 关注 ， 因 而 耐心 驻 
足 ， 进 而 增加 网 站 的 访问 量 ， 提 高 网 站 的 知名 度 。 

网 页 计数 器 模块 主要 从 计数 器 功能 实现 的 方法 着 手 ， 详 细 地 介绍 了 简单 的 数字 计数 器 、 图 形 数字 
计数 器 、GD2 图 形 数字 计数 器 、 数 据 库 数 字 计 数 器 的 实现 方法 。 


网 页 计数 器 的 操作 流程 如 图 2.1 所 示 。 


判断 登录 标 
记 是 否 为 空 


重新 输出 将 数据 添加 
网 站 访问 量 到 数据 库 中 


图 2.1 网 页 计数 器 的 操作 流程 
2.2 简单 数字 计数 器 


数字 计数 器 是 最 简单 的 一 种 ， 它 将 数据 存储 于 文本 文件 中 。 其 思路 是 首先 判断 文本 文件 是 否 存在 ， 如 
不 存在 ， 则 打开 失败 ;如 打开 成 功 ， 则 继续 执行 并 读 取 文 件 中 的 数据 ， 将 计数 器 增加 1。 然 后 以 写 的 方式 
重新 打开 文件 ， 把 新 的 统计 数据 写 入 到 文件 中 ,关闭 文件 。 最 后 重新 打开 文件 ， 读 取 并 输出 文件 中 的 数据 。 
简单 数字 计数 器 的 操作 流程 如 图 2.2 所 示 。 


判断 文件 
是 否 存在 


读 取 文 


图 2.2 简单 数字 计数 器 的 操作 流程 
按 上 述 原理 设计 的 计数 器 在 页 面 刷新 时 计数 器 的 值 就 会 增加 ， 所 以 要 想 让 页 面 刷新 时 计数 器 的 数 
据 不 增加 ， 最 好 是 应 用 Session 变量 防止 重复 计数 。 
® 
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当 应 用 Session 防止 重复 计数 后 , 无 论 如 何 刷 新 页 面 计数 器 的 值 都 不 会 增加 。 只 有 在 重新 打开 页 面 
时 ， 计 数 器 的 值 才 会 继续 增加 。 简 单数 字 计 数 器 的 结果 如 图 2.3 所 示 。 


公司 理念 : 以 高 新 技术 为 依托 ， 战略 性 地 开发 具有 巨大 市 场 潜力 的 高 价值 的 产品 - 


公司 远景 : 成 为 狂 有 核 必 技 术 和 核心 产品 的 高 科技 公司 ， 在 某 些 领 域 具 有 领先 的 市 场地 位 - 
核心 价值 观 : 永 葆 创业 激情 、 每 一 天 禾 在 进步 、 容 忍 失败 ， 鼓 励 创新 、 充 分 信任 、 平 等 交流 - 
网 站 当前 访问 量 : 100 


CopyRights reserved 2006 吉林 省 明日 科技 有 限 公司 
客户 服务 邮箱 : mingriseft@ningriseft con 


图 2.3 简单 数字 计数 器 的 结果 


分 析 简 单 的 数字 计数 器 ， 首 先 通过 session_start0 函 数 初始 化 一 个 Session 变量 。 通 过 Session 变量 
来 控制 页 面 刷 新 对 计数 器 的 影响 。 

然后 通过 让 语句 判断 Session 变量 的 值 是 否 为 空 ， 如 果 为 空 ， 则 打开 指定 的 文本 文件 ， 否 则 将 不 执行 。 

如 果 Session 变量 的 值 为 空 , 则 应 用 fopen() 函 数 以 只 读 的 方式 打开 指定 的 文本 文件 ,如果 文 件 不 存 
在 ， 则 输出 打开 文件 失败 ， 否 则 将 继续 执行 读 取 文件 中 的 数据 。 这 里 将 指定 的 文本 文件 counter.txt 存 
储 于 根 目 录 下 。 如 果 文 本 文件 存在 ， 则 继续 执行 ， 将 应 用 fgets() 函 数 读 取 文 本 文件 中 的 数据 ， 并 通过 
felose() 函 数 关闭 文件 。 此 时 将 读 取 到 的 数据 $counter 加 1， 即将 计数 器 中 的 值 加 1。 通 过 fopen() 函 数 重 
新 打开 指定 的 文本 文件 ， 注 意 是 以 写 入 的 方式 打开 ， 并 应 用 fputs() 函 数 将 新 统计 的 数据 写 入 到 文本 文 
件 中 ， 关 闭 文件 。 

最 后 为 Session 变量 赋值 ， 将 Session 变量 赋值 为 1， 用 于 控制 因 网 页 刷新 而 导致 计数 器 的 值 无 限 
增加 。 

简单 计数 器 的 文件 存储 于 TM\02\01\index.php 文件 中 。 关 键 代码 如 下 : 


<?php 

// 使 用 文本 存储 数据 

ifltisset($_SESSION[temp])){ // 判 断 $_SESSION[temp]=="" 的 值 是 否 为 空 ， 其 中 的 temp 为 自 定义 变量 
if(($fp=fopen("counter.txt","r"))==false){ 


echo "打开 文件 失败 1"; 

}else{ 

$counter=fgets($fp,1024); // 读 取 文 件 中 的 数据 
fclose($fp); 1/ 关闭 文本 文件 
$counter++; // 计 数 器 增加 1 
$fp=fopen("counter.txt","w"); /以 写 的 方式 打开 文本 文件 
fputs($fp,$counter); // 将 新 的 统计 数据 增加 1 
fclose($fp); 

/关闭 文件 
$_SESSION[temp]=1; // 计 数 器 的 值 增加 后 ， 为 $_SESSION[temp] 赋 值 1 
上 

// 从 文本 文件 中 读 取 统计 数据 

if(($fp=fopen("counter.txt","r"))==false){ 

echo "打开 文件 失败 1"; 

}else{ 

$counter=fgets($fp,1024); 


全 
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fclose($fp); 
} 


?> 
(1) fopen0 函 数 用 于 打开 某 文件 ， 并 返回 该 文件 的 标识 指针 。 该 函数 的 语法 如 下 : 
resource fopen(string filename, string mode [, int use_include_path [, resource context]]) 
fopen() 函 数 的 参数 说 明 如 表 2.1 所 示 。 
表 2.1 fopen() 函 数 的 参数 说 明 


参数 说 明 
filename | 必要 参数 。 用 于 指定 要 打开 文件 的 本 地 或 远程 地 址 
mode | 必要 参数 。 用 于 指定 要 打开 文件 的 模式 


可 选 参数 。 如 果 将 该 参数 设置 为 tue，PHP 会 尝试 按照 include_path 标准 包含 路 径 中 的 每 
个 指向 去 打开 文件 
可 选 参数 。 设 置 提 高 文件 流 性 能 的 一 些 选 项 


use_include_path 


context 


在 fopen() 函 数 的 mode 参数 中 ， 其 指定 的 打开 模式 很 多 ， 其 中 常用 的 有 如 下 儿 种 : 

r: 只 读 方式 打开 ， 将 文件 指针 指向 文件 头 。 

r+: 读 写 方式 打开 ， 将 文件 指针 指向 文件 头 。 

w: 写 入 方式 打开 , 将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文件 不 存在 则 尝试 创建 。 
w+: 读 写 方式 打开 ， 将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文 件 不 存在 则 尝试 创建 。 
(2) fgets() 函 数 用 于 获取 文件 指针 所 在 行 指定 长 度 的 内 容 。 该 函数 的 语法 如 下 : 


加 图 图 加 


string fgets(int handle Lintlength]) 


handle: 必要 参数 。 指 应 用 fopen() 或 fockopen0 函 数 成 功 打开 文件 或 连接 上 服务 器 后 所 返回 的 
文件 标识 。 
length: 可 选 参数 。 用 于 指定 读 取 一 行 最 多 的 字 节 数 ， 并 返回 length-1 个 字 节 的 字符 串 。 如 果 
省 略 该 参数 ， 则 length 的 长 度 默 认为 1024 个 字 节 。 
(3) fputs0 函 数 是 fwrite() 函 数 的 别名 ， 用 于 将 字符 串 写 入 到 指定 的 文件 中 ， 并 可 以 规定 写 入 字 节 
的 大 小 。 该 函数 的 语法 如 下 : 


int fwrite( resource handle, string string [, int length]) 


fputs() 函 数 的 参数 说 明 如 表 2.2 所 示 。 
表 2.2 fputs() 函 数 的 参数 说 明 


参 。 数 说 了 明 
handle 必要 参数 。 文 件 标识 指针 
sting ”| 。 必要 参数 。 要 写 入 某 文件 的 字符 串 


length 可 省 参数 。 指 写 入 文件 的 长 度 ， 如 果 省 略 该 参数 ， 则 将 指定 字符 串 的 所 有 内 容 写 入 到 文件 中 


至 此 ， 简 单 的 数字 计数 器 的 计数 部 分 设计 完成 ， 接 下 来 要 做 的 就 是 在 网 页 中 重新 输出 文本 文件 中 


@ 
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的 数据 ， 即 输出 网 站 当前 的 访问 量 。 代 码 如 下 : 


<?php 

// 从 文本 文件 中 读 取 统计 数据 
if(($fp=fopen("counter.txt","r"))==false}{ // 打 开 指 定 的 文件 
echo "打开 文件 失败 !"; 

jelsef 

$counter=fgets($fp,1024); // 读 取 文 件 内 容 


fclose($fp); 
echo "数字 计数 器 : " .$counter ; /输出 访问 次 数 


} 
?> 


2.3 图形 数字 计数 器 
所 谓 图 形 数 字 计 数 器 ， 就 是 将 计数 器 中 的 数据 以 数字 图 形 的 形式 显示 出 来 ， 其 设计 的 原理 与 简单 


数字 计数 器 是 相同 的 ， 唯一 不 同 的 地 方 就 是 在 对 数据 的 输出 过 程 中 ,数字 计数 器 直接 将 数据 进行 输出 ， 
而 图 形 数字 计数 器 则 将 数据 转化 成 数字 图 形 的 形式 进行 输出 。 其 运行 结果 如 图 2.4 所 示 。 


明日 回 尊 宫 理 系统 用: pe 
ye 和 3 修改 密码 
询 


您 是 第 [CHOHOHDHDHDhY 沪 问 本 系统 的 人 1! 简单 查阅 高 级 查 分 组 统计 退出 系统 


吉林 省 明日 科技 有 限 公司 是 一 家 以 计算 机 软件 技术 为 核心 的 高科 技 型 企业 ， 公 司 外 建 于 2000 年 12 月 ， 是 专业 的 应 
用 软件 开发 商 和 服务 提供 商 。 多 年 来 始终 致力 于 行业 管理 软件 开发 、 数 字 化 出 版 物 开发 制作 、 计 算 机 网 络 系统 媒 合 
图 2.4 图 形 数字 计数 器 
计数 器 功能 的 实现 方法 已 经 在 简单 数字 计数 器 中 进行 了 详细 的 讲解 ， 这 里 不 再 袭 述 。 这 里 主要 讲 
解 如 何 将 文本 文件 中 的 数据 以 图 形 的 形式 进行 输出 。 

图 形 数字 计数 器 存储 于 mm02\02\ 文 件 夹 下 ， 其 实现 关键 步骤 如 下 : 
首先 ， 是 读 取 文 本 文件 中 的 数据 ， 将 文本 文件 中 的 数据 写 入 到 一 个 变量 中 。 代 码 如 下 : 
<?php session_start(); 


ifttisset($_SESSION[temp])) // 判 断 $_SESSION[temp]==" 的 值 是 否 为 空 ， 其 中 的 temp 为 自 定义 变量 
if(($fp=fopen("counter.txt","r"))==false){ 


echo "打开 文件 失败 1"; 

Jelse{ 

$counter=fgets($fp,1024); // 读 取 文 件 中 的 数据 
fclose($fp); /关闭 文本 文件 
$counter++; 1/ 计数 器 增加 1 
S$fp=fopen("counter.txt","w"); /以 写 的 方式 打开 文本 文件 
fputs($fp,$counter); /将 新 的 统计 数据 增加 1 
fclose($fp); 


@ 


3 
$_SESSION[temp"]=1; 
} 


5 


第 2 章 网 页 计数 器 模块 (Apache+PHP+MySQL 5.0 实现 ) 


/| 为 $_ SESSION[temp] 赋 一 个 值 


然后 ， 将 读 取 的 数据 以 图 形 的 形式 进行 输出 ， 这 里 设计 的 是 一 个 6 位 数字 的 图 形 计数 器 ， 将 0 一 9 
的 数字 图 形 存储 在 根 目录 下 的 images 文件 夹 中 。 步 骤 如 下 : 


(1) 通过 strlen0 函 数 获取 数据 的 长 度 。 


(2) 输出 6 位 数字 的 图 形 计数 器 中 的 空 值 ， 即 处 理 6 位 数字 中 补 位 值 0 的 输出 。 首 先 ， 应 用 str_ 
repeat() 函 数 获取 6 位 数字 中 补 位 值 0 的 个 数 ， 然 后 应 用 for 语句 ， 根 据 补 位 值 0 的 个 数 〈S$str) ， 循 环 


输出 代表 0 的 图 形 。 


(3) 将 文本 文件 中 存储 的 数据 以 图 形 的 形式 输出 。 应 用 for 循环 语句 ， 根 据 strlen0) 函 数 获取 变量 
S$counter 的 长 度 ， 循 环 输出 指定 的 图 形 ， 应 用 switch 语句 为 数据 指定 不 同 的 图 形 。 


$len=strlen($counten); 


$str=str_repeat("0",6-$len); 
for($i=0;$i<strlen(S$str);$i++X{ 


$result=$str[$i]; 


$result='<img src=images/0.gif>"; 


echo $result; 


} 
for($i=0;$i<stren($counter);$i++}{ 


$result=$counter[$i]; 
switch($resultX{ 


// 如 果 值 为 0， 则 输出 0.gif 图 片 


case "0"; $ret[$i]="0.gif"; 
case "1"; $ret[$i]="1.gif"; 
case "2"; $ret[$1]="4.gif"; 
case "3"; $ret[$i]="3.gif"; 
case "4"; $ret[$i]="4.gif"; 
case "5"; $ret[$1]="5.gif"; 
case "6"; $ret[$i]="6.gif"; 
case "7"; $ret[$1]="7.gif"; 
case "8"; $ret[$1]="4.gif"; 
case "9"; $ret[$i]="9.gif"; 


echo "<img src=images/".$ret[$i].".>"; 


vv 一 一 


break; 
break; 
break; 
break; 
break; 
break; 
break; 
break; 
break; 
break; 


// 获 取 字 符 串 的 长 度 
/| 获取 6-$len 个 数字 0 
// 获 取 变 量 $str 的 字符 串 长 度 


/| 循环 输出 $result 的 结果 
// 获 取 字 符 串 的 长 度 


// 获 取 数 据 
// 为 不 同 的 数据 指定 不 同 的 图 形 


/| 输出 文本 文件 中 存储 的 数据 


说 明 @ strlen0) 函 数 用 于 获取 指定 字符 囊 的 长 度 . 
@ str_ repeat0 函 数 用 于 将 指定 的 字符 串 重复 输出 。 
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2.4 GD2 图 形 计数 器 


GD2 图 形 计 数 器 将 计数 器 中 存储 的 数据 以 GD2 函数 绘制 图 形 的 形式 进行 输出 。 其 运行 结果 如 图 2.5 
所 示 。 
全 业 精 神 : 博学 、 创 新 、 求 实 、 竺 行 
公司 理念 : 以 商 新 技术 为 依托 ,战略 性 地 开发 具有 巨大 市 场 洪 力 的 高 价值 的 产品 
公司 远 县 : 成 为 期 有 核心 技术 和 核心 产品 的 高 科技 公司 , 在 某 些 领 域 具有 领先 的 市 场地 位 -。 
核心 价值 观 : 永 葆 创业 激情 、 每 一 天 和 在 进步 、 容 忍 失败 ， 鼓励 创新 、 充 分 信任 、 平 等 交流 - 


网 站 护 访 癌 量 66 


© CopyBights reserved 2008 吉林 省 明日 科技 有 限 公司 
客户 服务 邮箱 : ningrisoftBningrisoft. com 


图 2.5 GD2 图 形 计数 器 


计数 器 中 的 数据 仍然 存储 在 文本 文件 中 ， 至 于 如 何 存储 和 读 取 其 中 的 数据 这 里 不 再 歼 述 ， 直 接 讲 
解 如 何 将 读 取 的 数据 以 GD2 图 形 的 形式 进行 输出 。GD2 图 形 计数 器 存储 于 mr\02\03 文件 来 下 ， 其 关 
键 实现 步骤 如 下 : 

首先 ， 创 建 一 个 画布 ， 并 指定 画布 背景 的 颜色 和 字体 的 颜色 。 通 过 imagecreate0 函 数 创建 画布 ， 应 
用 imagecolorallocate() 函 数 来 分 别 定 义 背 景 和 字体 的 颜色 ， 其 中 在 定义 字体 的 颜色 时 应 用 randO) 函 数 ， 
通过 其 在 指定 的 范围 内 随机 选择 颜色 。 

然后 ， 应 用 iconv() 函 数 对 指定 的 中 文字 符 进 行 转换 ， 并 应 用 imagettftext() 函 数 输出 中 文字 符 串 。 

最 后 ， 应 用 imagestring() 函 数 输出 计数 器 中 存储 的 数据 ， 并 应 用 imagepng0 函 数 以 PNG 格式 将 图 
像 输出 到 浏览 器 或 文件 中 ， 通 过 imagedestroy(0) 函 数 销 毁 图 像 。 代 码 如 下 : 


<?php 
if(($fp=fopen("counter.txt","r"))==false){ 
echo "打开 文件 失败 六; 
}else{ 
$counter=fgets($fp,1024); 
fclose($fp); 
// 通 过 GD2 函数 创建 画布 
$im=imagecreate(240,24); 
$gray=imagecolorallocate($im,255,255,255); /定义 背景 
$color =imagecolorallocate($im,rand(0,255),rand(0,255),rand(0,255)); /定义 字体 颜色 
/输出 中 文字 符 
$text=iconv("gb2312","utf-8"," 网 站 的 访问 量 : "); // 对 指定 的 中 文字 符 串 进行 转换 
S$font = "fonts/FZHCJW.TTF"; 


imagettftext($im,14,0,20, 18,$color,$font, $text); // 输 出 中 文 
/输出 网 站 的 访问 次 数 
imagestring($im,5,160,5,$counter$color); 
imagepng($im); 
imagedestroy($im); /以 PNG 格式 将 图 像 输出 到 浏览 器 
} 
a 
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(1) imagecreate() 函 数 用 于 创建 一 个 新 的 图 像 。 该 函数 的 语法 如 下 : 


resource imagecreate( int x_size, inty_size ) 


该 函数 用 于 返回 一 个 图 像 标 识 符 ， 参 数 x_size、y_size 为 图 像 的 尺寸 ， 单 位 为 像素 (pixel) 。 
(2) imagecolorallocate() 函 数 返 回 一 个 标识 符 ， 代 表 了 由 给 定 的 RGB 成 分 组 成 的 颜色 。 该 函数 的 
语法 如 下 : 


int imagecolorallocate( resource image, int red, int green, int blue) 


回 “image 参数 是 imagecreatetruecolor() 函 数 的 返回 值 。 
red、green 和 blue 分 别 是 所 需要 颜色 的 红 、 绿 和 蓝 成 分 。 这 些 参数 是 0 一 255 的 整数 或 者 十 六 
进 制 的 0x00 一 0xFF。 
imagecolorallocate() 函 数 必须 被 调用 以 创建 每 一 种 用 在 image 所 代表 的 图 像 中 的 颜色 。 第 一 次 对 
imagecolorallocate() 函 数 的 调用 会 填充 背景 色 。 
(3) rand(0) 函 数 产生 一 个 随机 数 ， 返 回 随机 数 的 值 。 该 函数 的 语法 如 下 : 


int rand( int min, int max) 


如 果 没 有 提供 可 选 参数 min 和 max, 则 返回 0 一 RAND_MAX 之 间 的 伪 随 机 数 ; 如 果 提 供 可 选 参数 ， 
则 返回 可 选 参数 范围 之 内 的 随机 数 。 
(4) iconv(0) 函 数 用 于 将 一 种 已 知 的 字符 集 文件 转换 成 另 一 种 已 知 的 字符 集 文件 。 该 函数 的 语法 如 下 : 


iconv( string in_charset, string out_charset, string str) 


(5) imagettftext0 函 数 以 TTF (TrueType Fonts) 字体 向 图 像 中 绘制 字符 串 。 该 函数 的 语法 如 下 : 


array imagettftext( resource image, float size, float angle, int x, int y, int color, string fontfile, string text ) 
imagettftext() 函 数 的 参数 说 明 如 表 2.3 所 示 。 
表 2.3 imagettftext() 函 数 的 参数 说 明 


参数 说 明 
image 图 像 资源 
size 字体 大 小 。 根 据 GD 版 本 不 同 ， 应 该 以 像素 大 小 “GD1) 或 点 大 小 〈GD2) 指定 


i 0 顺 时 针 计 算 ，0 度 为 水 平 ， 也 就 是 3 点 钟 的 方向 〈 由 左 到 右 ) ，90 度 则 为 由 下 到 上 的 
和 文字 的 x 坐标 。 设 定 第 一 个 字符 的 基本 点 

文字 的 y 坐 标 。 设 定 字体 基线 的 位 置 ， 不 是 字符 的 最 底 端 

color 文字 的 颜色 

fontfile 字体 的 文件 名 称 ， 也 可 以 是 远 端的 文件 

text 字符 串 内 容 


(6) imagestring() 函 数 实现 水 平 的 绘制 一 行 字符 串 。 语 法 如 下 : 


bool imagestring( int im, int font, int x, int y, string str, int col ) 
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imagestring() 函 数 用 col 颜色 将 字符 串 str 绘制 到 im 所 代表 的 图 像 的 (x，y) 坐标 处 〈 这 是 字符 串 
左上 角 坐 标 ， 整 幅 图 像 的 左上 角 为 《0，0) ) 。 如 果 font 是 1、2、3、4 或 5， 则 使 用 内 置 字体 。 

至 此 ，GD2 图 形 计数 器 讲解 完毕 。 上 述 介绍 的 这 3 种 计数 器 都 是 从 输出 的 方式 上 进行 区 别 的 ， 其 
统计 的 方式 是 完全 相同 的 ， 下 面 将 要 继续 介绍 几 种 完全 不 同 的 计数 器 。 


2.5 ”数据 库 数字 计数 器 


数据 库 计数 器 ， 顾 名 思 义 ， 就 是 将 统计 的 数据 存储 到 数据 库 中 ， 不 但 可 以 统计 出 一 个 数字 值 ， 而 
且 还 可 以 记录 更 多 的 信息 例如 ，IP 地 址 、 访 问 时 间 等 ) ， 这 些 都 可 以 存储 在 数据 库 中 。 数 据 库 计 数 
器 相对 于 其 他 几 种 方法 而 言 ， 它 的 功能 是 最 为 强大 的 。 其 运行 结果 如 图 2.6 所 示 。 


NL 


一 一 网 站 的 访问 量 : 简单 查询 分 姐 统 计 退出 系统 


言 林 省 明日 科技 有 限 公司 是 一 家 以 计算 机 软件 技术 为 核心 的 高 科技 型 企业 ,公司 创建 于 2000 年 12 月 ,是 专业 的 应 
用 软件 开发 商 和 服务 提供 商 。 多 年 来 始终 致力 于 行业 管理 软件 开发 、 数 字 化 出 版 物 开发 制作 、 计 算 机 网 络 系统 综合 
全 业 宗 虽 : 为 企业 服务 ， 打 造 企业 智 能 管理 平台 ,改善 企业 的 管理 与 运作 过 程 ,提高 企业 效率 , 降低 管理 成 本 ， 增 
核心 价值 观 : 水 葆 创业 激情 、 每 一 天 都 在 进步 、 容 忍 失 败 ,就 励 创新 、 充 分 信任 、 平 等 交流 。 

统计 dp 的 自重 : 2 
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图 2.6 数据 库 数字 计数 器 
既然 数据 库 计 数 器 需要 数据 库 的 支持 ， 所 以 首先 要 创建 一 个 数据 库 ， 并 在 该 数据 库 中 创建 一 个 数 
据 表 用 于 存储 数据 。 创 建 的 数据 表 如 图 2.7 所 示 。 
胃 服务 器 : localhost 》 忆 数据 库 : db_counter ， 国 表 :tb_count04 
辕 训 览 ， 阿 结构 ,内 SQL 亡 搜 索 ， 子 项 入 联 导 出 “上 阿 Imporl， 做 换 作 [而 清空 大 到 除 


类 型 额外 
| 出 int(10) auto_increment 围 
到 varchar(50) -gb2312_chinese_ci 
广 Varchar(50) gb2312_chinese_ci 
记 varchar(50) gb2312_chinese_ci 
varchar(50) gb2312_chinese_ci 


7 
2 
2 
pd 
办 


图 2.7 创建 的 数据 表 
然后 连接 数据 库 ， 连 接 数据 库 文件 (mr02\04\connvconn.php) 的 代码 如 下 : 


<?php 

$id=mysql_connect(localhost,root,"111");// 连 接 数 据 库 服务 器 
mysql_select_db(db_counter,$id); /| 连接 db_counter 

mysql_query("set names gb2312"); // 指 定数 据 库 中 字符 串 的 编码 格式 为 gb2312 
?> 


@ 
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a Ce 
库 要 正确 ， 以 及 对 数据 库 中 字符 囊 编码 格式 的 转 摘 . 
最 后 实现 网 站 访问 信息 的 存储 ， 并 且 将 访问 量 和 IP 访问 量 以 图 形 和 图 像 的 形式 进行 输出 。 数据库 
数字 计数 器 存储 于 TM\02\04\ 文 件 夹 下 ， 其 关键 实现 步骤 如 下 : 
(1) 初始 化 一 个 Session 变量 ， 与 数据 库 建立 连接 ,获取 当前 时 间 和 客户 端的 IP 地 址 ， 并 判断 Session 
变量 的 值 是 否 为 宝 。 代 码 如 下 


<?php 

Session_start(); 

include("conn/conn.php"); 

$data1=date("Y-m-d h:m:s"); // 获 取 当 前 访问 时 间 

$data2=date("Y-m-d"); /获取 当前 访问 时 间 

S$ip=getenv('REMOTE_ADDR'); // 获 取 客 户 端的 IP 地 址 

if($_SESSION[temp]==""){ /| 判断 $_SESSION[temp]=="" 的 值 是 否 为 空 ， 其 中 的 temp 为 自 定义 变量 


(2) 通过 数据 库存 储 网 站 被 访问 的 数据 ,执行 select 查询 ， 判 断 当前 获取 的 IP 地 址 在 数据 库 中 是 
和 否 已 经 存在 。 如 果 数 据 库 中 存在 该 PP 地 址 ， 则 执行 update 更 新 语句 ， 将 该 IP 地 址 对 应 的 数据 的 访问 
量 增 加 1， 并 且 更 新 访问 的 时 间 ;， 如果 数 据 库 中 不 存在 该 IP 地址， 则 执行 insert 添加 语句 ， 将 该 IP 地 
址 的 数据 添加 到 数据 库 中 。 代 码 如 下 : 


$query=mysql_query("select * from tb_count04 where ip='$ip"); 

$result=mysql_num_rows($query); 

if($result>0){ 

$query_1=mysql_query("update tb_count04 set counts=counts+1,data1='$data1',data2='$data2' where ip='$ip"); 
}else{ 

$query="insert into tb_count04(counts,data1,data2,ip)values('1','$data1','$data2','$ip’)"; 
$result=mysql_query($query); 

} 


(3) 将 Session 变量 赋值 为 1， 防止 因 页 面 刷 新 导致 重复 计数 。 代 码 如 下 : 
$_SESSION[temp]=1; /登录 以 后 ，$_SESSION[temp] 的 值 不 为 空 ， 给 $_SESSION[temp] 赋 值 1 
: 


人 
至 此 ， 一 次 网 站 访问 量 的 统计 结束 。 
2.6 网 站 访问 量 统计 分 析 
计数 器 中 的 有 些 功 能 已 经 在 前 面 的 章节 中 讲解 过 了 ， 例 如 ， 采 用 数据 库 来 统计 网 站 的 访问 量 ， 通 


过 Session 来 防止 网 站 的 重复 计数 ， 应 用 图 形 数字 计数 器 来 输出 网 站 当前 的 访问 量 。 下 面 介绍 如 何 对 网 
站 访问 量 的 数据 进行 统计 分 析 。 这 里 主要 应 用 Jpgraph 类 创建 柱 形 图 来 对 网 站 访问 量 的 结果 进行 分 析 ， 


@ 
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。 二 
其 运行 效果 如 图 2.8 所 示 。 
只 查看 访问 是 
网 站 总 的 访问 重 : 177 有 Wa 是: 1 网 站 当日 的 访问 量 : 
网 站 总 的 IP 访 问 重 : 1 当月 的 访问 量 : 1 网 站 当日 JP 的 访问 量 : 0 
0-0 


Y 2011-o4 月 访问 和 走势 名 


Counter 
15 


2011-04 月 网 站 访问 时 走势 到 


2.8 ”对 网 站 访问 量 的 结果 进行 分 析 
2.6.1 通过 数据 库 记 录 网 站 访问 量 


有 关 数 据 库 统计 网 站 访问 量 的 具体 方法 已 经 在 2.5 节 中 进行 了 详细 讲解 , 这 里 不 再 獒 述 。 输 出 数据 
库 中 存储 的 网 站 访问 量 应 用 的 是 2.3 节 的 图 形 数字 计数 器 方法 。 其 关键 代码 如 下 : 


<?php 

// 以 图 形 的 形式 输出 数据 库 中 的 记录 数 

$querys="select sum(counts) as Ilfrom tb_count10 "; /查询 数据 库 中 总 的 访问 量 
$results=mysql_query($querys,$id); 

$fwl=mysql_result($results,0,); 


echo "- 一 一 一 一 - 

// 对 补 位 数字 0 的 处 理 

$len=strlen($fwl); /获取 字符 串 的 长 度 
$str=str_repeat("0",6-$len); /获取 6-$len 个 数字 0 
for($i=0;$i<strlen($stn;$i++)f /获取 变量 $str 的 字符 串 长 度 


$resultes=$str[$i]; 
$resultes='<img src=images/0.gif>"; 


echo $resultes; /循环 输出 $resultes 的 结果 
} 

// 对 数据 库 中 数据 的 处 理 

for($i=0;$i<strlen($fwl):$i++){ /获取 字符 串 的 长 度 
$result=$fwl[$i]; 

Switch($result)f 


/如果 值 为 0， 则 输出 0.gif 图 片 
case "0"; $ret[$i]="0.gif";breaki; 
case "1"; $ret[$i]="1.gif";breaki; 
case "2"; $ret[$i]="2.gif";breaki; 
case "3"; $ret[$i]="3.gif";breaki; 
case "4"; $ret[$i]="4.gif";breaki 
case "5"; $ret[$i]="5.gif";breaki; 


@_ 
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case "6"; $ret[$i]="6.gif";break; 
case "7"; $ret[$1]="7.gif";break; 
case "8"; $ret[$1]="8.gif";break; 
case "9"; $ret[$i]="9.gif";break; 


echo "<img src=images/". $ret[$1].".>"; /| 输出 访问 次 数 


» 
?> 


网 站 访问 量 的 输出 结果 如 图 2.9 所 示 。 


一 -一 一 [HCHOHDHEMHEY Fa 是 统计 分 析 


2.9 输出 网 站 访问 量 
2.6.2 ”网 站 访问 量 统计 分 析 


在 图 2.9 的 网 站 访问 量 输出 结果 中 ， 有 一 个 “网 站 访问 量 统计 分 析 ” 超 链接 ， 它 链接 到 indexs.php 
文件 中 ， 在 此 文件 中 完成 对 网 站 访问 量 的 统计 操作 ， 其 统计 的 内 容 如 图 2.10 所 示 。 


Am 
ma - wre. mn ar TCL 
只 查看 访问 量 
网 站 总 的 访问 量 : 177 网 站 当月 的 访问 量 : 1 网 站 当日 的 访问 量 : 
网 站 总 的 IP 访 问 量 : 1 网 站 当月 I 的 访问 量 : 1 网 站 当日 IP 的 访问 量 : 0 
zm > 国 到 


图 2.10 网 站 访问 量 统计 
(1) 连接 数据 库 ， 调 用 指定 的 数据 库 连 接 文 件 。 代 码 如 下 : 


<?php include("conn/conn.php"); ?> 


(2) 输出 网 站 总 的 访问 量 以 及 总 的 JP 访问 量 。 应 用 select 语句 和 sum0 函 数 ， 从 数据 表 中 读 取 counts 
字段 总 的 数据 ， 获 取 网 站 总 的 访问 量 。 代 码 如 下 : 


<td width="240" height="25" align="center" valign="middle"> 
<?php 

$query_1="select sum(counts) as countes from tb_count10 "; 
// 查 询 数据 库 中 总 的 访问 量 

$result_1=mysql_query($query_1); 
$countes_1=mysql_result($result_1,0, countes'); 

echo "<p class='STYLE1'> 网 站 总 的 访问 量 : $countes_1</p>"; 
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(3) 获取 网 站 总 的 耳 访 问 量 。 从 数据 表 中 读 取 数 据 ， 应 用 mysql_fetch_array() 函 数 获 取 ip 字段 对 
应 的 IP 地 址 ， 并 将 获取 到 的 值 赋 给 一 个 数组 ， 通 过 array_unique() 函 数 去 除数 组 中 重复 的 值 ， 最 终 获 取 
到 网 站 总 的 他 访问 量 。 代 码 如 下 : 
// 查 询 数 据 库 中 总 的 IP 访问 量 
$query_2="select * from tb_count10 "; 


$result 2=mysql_query($query_2); 
while($myrow=mysql_fetch_array($result_2)){f 


$counts_2[=$myrow[ip1]; /将 获取 的 ip 值 赋 给 变量 

} 

$results_2=array_unique($counts_2); // 去 除数 组 中 重复 的 值 
$countes_2=count($results_2); // 获 取 数 组 中 值 的 数量 ， 即 总 的 1P 访问 量 
echo "<p class='STYLE1'> 网 站 总 的 IP 访问 量 : $countes_2</p>"; 

?></td> 


(4) 获取 网 站 当月 的 访问 量 和 当月 的 IP 访 问 量 。 使 用 的 方法 与 获取 网 站 总 的 访问 量 和 总 的 IP 访 
问 量 的 方法 是 相同 的 。 区 别 之 处 是 select 语句 中 增加 了 一 个 查询 条 件 “data2=".substr(date("Y-m-d")， 
0,7)."”， 即 查询 数据 库 中 符合 当前 月 份 的 数据 。 代 码 如 下 : 


<td width="240" align="center" valign="middle"> 

<?php 

// 查 询 数据 库 中 当月 总 的 访问 量 

$query_3="select sum(counts) as countes from tb_count10 where data2=".substr(date("Y-m-d"),0,7)."™"; 
$result_3=mysqlL_query($query_3); 
$countes_3=mysql_result($result_3,0,countes'); 

echo "<p class='STYLE2'> 网 站 当月 的 访问 量 : $countes_3</p>"; 

// 查 询 数据 库 中 当月 的 IP 访问 量 

$query_4="select * from tb_count10 where data2=".substr(date("Y-m-d"),0,7)."™"; 
Sresult_ 4=mysql_query($query_4); 

$counts_4=array(); 

while($myrow_4=mysql_fetch_array($result_4)){ 


$counts_40=$myrow_4[ip"; // 将 获取 的 ip 值 赋 给 变量 

} 

$results_4=array_unique($counts_4); // 去 除数 组 中 重复 的 值 
$countes_4=count($results_4); // 获 取 数 组 中 值 的 数量 ， 即 总 的 1P 访问 量 
echo "<p class='STYLE2'> 网 站 当月 IP 的 访问 量 : $countes_4</p>"; 

?></td> 


(5) 获取 网 站 当日 的 访问 量 和 当日 的 IP 访问 量 。 其 使 用 的 方法 这 里 不 再 袭 述 ， 指 定 的 查询 条 件 
是 “datal=".date("Y-m-d")."”。 程 序 关键 代码 如 下 : 


<td width="248" align="center" valign="middle"> 

<?php 

// 查 询 数据 库 中 当日 总 的 访问 量 

$query_5="select sum(counts) as countes from tb_count10 where data1=".date("Y-m-d").™" "; 
$result_5=mysqlL_query($query_5); 

$countes_5=mysql_result($result_5,0,countes'); 

echo "<p class='STYLE3'> 网 站 当日 的 访问 量 : $countes_5</p>"; 


@ 
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// 查 询 数据 库 中 当日 的 IP 访问 量 

$query_6="select * from tb_count10 where data1=".date("Y-m-d")."™"; 
$result_ 6=mysql_query($query_6); 

$counts_6=array(); 

while($myrow_6=mysql_fetch_array($result_6)){ 


$counts_6[0=$myrow_6[ip1]; // 将 获取 的 ip 值 赋 给 变量 

} 

if(is_array($counts_6)X 

$results_6=array_unique($counts_6); /去 除数 组 中 重复 的 值 
$countes_6=count($results_6); // 获 取 数 组 中 值 的 数量 ， 即 总 的 IP 访问 量 
echo "<p class='STYLE3'> 网 站 当日 IP 的 访问 量 : $countes_6</p>"; 

} 

<d> 


2.6.3” 柱 形 图 分 析 网 站 月 访问 量 比重 


在 indexs.php 文件 中 ， 通 过 下 拉 列 表 框 选择 指定 的 月 份 ， 单 击 “ 提 交 ” 按 钮 ， 将 在 本 页 以 柱 形 图 
的 形式 输出 网 站 指定 月 份 的 访问 量 比重 。 
首先 创建 表单 ， 提 交 指 定 要 分 析 的 月 份 。 其 关键 代码 如 下 : 


<form name="form1" method="post" action="indexs.php"> 

<select name="select"> 

<?php 

$que="select distinct data2 from tb_count10"; /查询 数据 库 中 月 份 的 数据 
$res=mysql_query($que,$id); 

while($myrow=mysql_fetch_array($res)X{ // 循 环 输出 月 份 数据 

?> 

<option value="<?php echo $myrow['data2];?>"><?php echo $myrow['data2];?></option> 
<?php 

} 

?> 

</select>&nbsp;&nbsp; 

<input type="submit" name="Submit" value=" 提 交 "> 

</form> 


然后 在 本 页 中 获取 表单 提交 的 月 份 数据 ， 以 表单 提交 的 数据 为 条 件 进 行 查询 ， 将 从 数据 库 中 读 取 
的 数据 存储 到 指定 的 数组 中 ， 并 将 数组 重新 生成 新 字符 串 ， 将 新 字符 串 作为 超 链 接 的 参数 值 传递 到 
stat.php 文件 中 。 关 键 代 码 如 下 : 


<?php 

if(isset($_POST['select")){ // 淹 断 提交 的 数据 是 否 为 真 

$query="select counts,data1 from tb_count10 where data2=".$_POST[select]." order by data1 ";// 定 义 SQL 语句 
$result=mysql_query($query); /执行 查询 语句 

$results=array(); /定义 数组 

$datas=array(); /定义 数组 

while($myrow=mysql_fetch_array($result)){ // 循 环 输出 查询 结果 

$results]=$myrow['counts']; // 将 查询 结果 存储 到 数组 中 


@ 
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$Imbs=implode(",", $results); 
$datas[]=substr($myrow['data1"],8,2); 
$das=implode(",",$datas); 

} 

?> 


// 将 数组 重新 生成 一 个 字符 串 
// 截 取 时 间 数 据 ， 并 存储 到 数组 中 
// 将 数组 重新 生成 一 个 字符 串 


<table width="902" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor="#D5D5D5"> 


<tr> 


<td height="18" background="images/mysql_9.jpg" bgcolor="#FFFFFF" class="STYLE4"> 


<?php echo $_POST[select1];?> 月 网 站 访问 量 走势 图 
</td> 

</tr> 

<tr> 

<td bgcolor="#FFFFFF"> 


<p><img src="stat.php?Imbs=<?php echo $Imbs; ?>&das=<?php echo $das;?>"/></p> 
<p align="center" class="STYLE2"><?php echo $_POST[select];?> 月 网 站 访问 量 走势 图 </p> 


</td> 
</tr> 
</table> 
<?php 
} 


人 > 


最 后 ， 在 stat.php 文件 中 应 用 Jpgraph 类 库 创 建 柱 形 图 ， 以 超 链接 传递 的 参数 值 作为 数据 ， 生 成 网 


站 月 访问 量 统计 图 。stat.php 文件 的 代码 如 下 : 


<?php 

include ("src/jpgraph.php"); 

include ("src/jpgraph_bar.php"); 

$Imbs=$_GET[Imbs"; 

$das=$_GET[das']; 

$datay=explode(",",$Imbs); 

$datax=explode(",",$das); 

$graph = new Graph(800,200,"auto"); 
$graph->img->SetMargin(60,20,30,50); 
$graph->SetScale("textlin"); 
$graph->SetMarginColor("lightblue"); 
$graph->title->Set('counter); 
$graph->title->SetFont(FF_SIMSUN, FS_BOLD); 
$graph->xaxis->SetFont(FF_VERDANA,FS_NORMAL,10); 
$graph->yaxis->SetFont(FF_VERDANA,FS_NORMAL,10); 
$graph->xaxis->SetTickLabels($datax); 
$graph->xaxis->SetLabelAngle(50); 

$bplot = new BarPlot($datay); 

$bplot->SetWidth(0.6); 


// 载 入 类 库 文件 
// 获 取 超 链接 传递 的 参数 值 


/按照 “,” 分 隔 字符 串 ，$data 为 获取 的 数据 变量 
/按照 “,” 分 隔 字符 串 ，$data 为 获取 的 数据 变量 
// 创 建 图 像 

ll 设置 图 像 边框 间距 

// 定 义 坐 标 刻度 类 型 

/定义 图 像 颜色 

/定义 标题 

// 设 置 标题 字体 

// 设 置 X 轴 的 字体 

// 设 置 Y 轴 的 字体 

// 设 置 X 轴 输 出 的 数据 

// 设 置 输出 文字 大 小 

/实例 化 图 像 创建 类 

// 设 置 柱 形 图 的 输出 大 小 


$bplot->SetFillGradient("navy"," 示 FFFF00",GRAD_LEFT_REFLECTION); /设置 图 像 的 类 型 和 填充 颜色 


$bplot->SetColor("white"); 
$graph->Add($bplot); 
$graph->Stroke(); 

多 


@ 


// 设 置 图 像 边框 颜色 
// 添 加 数据 
// 生 成 图 像 
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2.7 技术 提炼 


网 页 计数 器 的 设计 方法 很 多 ， 应 用 不 同方 法 设计 的 计数 器 的 实效 性 也 不 尽 相同 。 下 面 对 网 页 计数 
器 开发 中 应 用 到 的 技术 进行 提炼 。 


2.7.1 防止 重复 计数 


在 开发 网 页 计数 器 的 程序 中 ， 一 个 最 为 关键 的 问题 就 是 如 何 防止 网 站 的 重复 计数 。 如 果 一 个 计数 
器 不 能 控制 重复 计数 ， 那 么 这 个 计数 器 将 没有 任何 意义 。 根 据 不 同 的 计数 器 的 设计 方法 ， 总 结 出 两 种 
控制 重复 计数 的 方法 。 

1. 通过 Session 防止 重复 计数 

该 方法 可 以 应 用 到 任意 一 个 计数 器 中 实现 控制 重复 计数 的 功能 。 其 实现 的 原理 是 : 在 当前 页 被 访问 
时 ， 初 始 化 一 个 Session 变量 ， 并 赋 给 其 一 个 空 值 ， 然 后 判断 Session 变量 的 值 是 否 为 室 ， 如 果 为 室 ， 则 
将 计数 器 的 值 增 加 1， 并 且 为 Session 变量 赋值 为 1。 此 时 ， 在 当前 页 中 ，Session 变量 的 值 已 经 不 为 空 ， 
无 论 如 何 刷新 ，Session 变量 的 值 都 不 会 改变 ， 所 以 计数 器 的 值 也 不 会 增加 。 关 键 代码 如 下 : 


<?php 

// 使 用 文本 存储 数据 

if(lisset($_SESSION[temp1)X /判断 $_SESSION[temp]==" 的 值 是 否 为 空 ， 其 中 的 temp 为 自 定义 变量 
if(($fp=fopen("counter.txt","r"))==false){ 

echo "打开 文件 失败 !"; 

}else{ 

$counter=fgets($fp,1024); // 读 取 文件 中 数据 

fclose($fp); /关闭 文本 文件 

$counter++; /计数 器 增加 1 

$fp=fopen("counter.txt","w"); /以 写 的 方式 打开 文本 文件 

fputs($fp,$counter); /将 新 的 统计 数据 增加 1 

fclose($fp); 

} /关闭 文本 文件 

$_SESSION[temp]=1; /1 计数 器 的 值 增加 后 ， 为 $_SESSION[temp] 赋 值 1 


// 从 文本 文件 中 读 取 统计 数据 
if(($fp=fopen("counter.txt","r"))==false){ 
echo "打开 文件 失败 !"; 

}else{ 

$counter=fgets($fp,1024); 

fclose($fp); 

上 


?> 
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WY a a 
值 仍然 会 增加 。 


2. 通过 IP 防止 重复 计数 

通过 IP 防止 重复 计数 是 最 准确 的 方法 ， 它 完全 是 通过 客户 端 机 器 的 IP 来 防止 重复 计数 ， 不 会 有 
任何 时 间 和 页 面 的 限制 ， 只 要 是 已 经 访问 过 的 IP， 计 数 器 的 值 就 不 会 增加 。 

通过 IP 防止 重复 计数 必须 要 有 数据 库 的 支持 ， 服 务 器 会 将 每 个 访问 过 本 网 站 的 客户 端的 IP 存储 
到 数据 库 中 。 

当 客 户 端的 机 器 访问 本 网 站 时 ， 服 务 器 将 会 首先 获取 该 客户 端的 IP 地 址 ， 并 且 与 数据 库 中 存储 的 
IP 地 址 进行 比较 ， 如 果 数 据 库 中 已 经 存在 此 IP， 则 计数 器 的 值 将 不 会 增加 ; 否则 将 计数 器 的 值 加 1， 
并 且 将 该 IP 地 址 存储 到 数据 库 中 。 关 键 代码 如 下 : 


<?php 
session_start(); 
include("conn/conn.php"); 


$data1=date("Y-m-d h:m:s"); /获取 当前 访问 时 间 
$data2=date("Y-m-d"); // 获 取 当 前 访问 时 间 
S$ip=getenv('REMOTE_ADDR'); /获取 客户 端 IP 地 址 
// 使 用 数据 库存 储 数据 


$query=mysql_query("select * from tb_count04 where ip='$ip"); 
$result=mysqlL_num_rows($query); 

if($result>0){ 

echo "该 IP 地 址 已 经 存在 "; 

}else{ 

$query="insert into tb_count04(counts, data1,data2,ip)values('1','$data1','$data2','$ip'")"; 
$result=mysql_query($query); 

} 


os 
通过 IP 不 但 可 以 防止 重复 计数 ， 而 且 可 以 统计 出 网 站 被 同一 卫 访 问 的 次 数 。 
2.7.2 通过 array_unique() 函 数 获 取 网 站 当月 IP 访问 量 


通过 array_unique() 函 数 去 除数 组 中 重复 的 值 ， 从 而 实现 统计 网 站 当月 IP 的 访问 量 。 其 中 主要 应 用 
了 array_unique() 函 数 和 count() 函 数 。 
(1) array_unique() 函 数 
该 函数 将 值 作 为 字符 串 排序 ， 然 后 对 每 个 值 只 保留 第 一 个 键 名 ， 忽 略 所 有 后 面 的 键 名 。 语 法 如 下 : 


array array_unique( array array) 
(2) count() 函 数 
该 函数 返回 数组 中 的 单元 数目 ， 用 来 计算 数组 中 值 的 个 数 。 语 法 如 下 : 


int count( mixed array [, int mode]) 


@ 
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参数 array 为 指定 的 数组 ;参数 mode 为 可 选 参 数 ， 如 果 mode 的 值 设置 为 COUNT_RECURSIVE 
(或 1) ， 该 函数 将 递归 地 对 数组 计数 。 对 计算 多 维 数组 的 所 有 单元 尤其 有 用 。 此 参数 的 默认 值 是 0。 
在 获取 网 站 当月 IP 的 访问 量 之 前 ， 先 来 看 一 下 存储 网 站 访问 量 数据 的 数据 表 的 结构 ， 如 图 2.11 


胃 服务 器 : localhost 加 数据 库 : db_counter ， 国 表 :tb_count10 
| 围 训 览 “ 因 结构 , 民 soL 万 搜 索 也 : 括 入 此 导出 ” 卫 Import 多 所 作出 清空 央 昌 除 
字段 类 型 整理 尾 性 Null 默认 额外 操作 
厂 也 int(l0) 理 auto_increment 凡 X 图 
厂 ”counts varchar(50) gb2312_chinese_ci 否 国 户 X 
厂 datat varchar(50) gb2312_chinese_ci 否 Ep Die 
厂 dataz varchar(50) gb2312_chinese_ci 否 Xx 
Fw varchar(50) gb2312_chinese_ci 理 FX 


图 2.11 网 站 访问 量 统计 数据 表 
在 该 数据 表 中 ， 包 括 5 个 字段 ， 分 别 是 id; 网 站 访问 量 counts; 访问 时 间 datal， 完 整 的 时 间 年 月 
日 ; 访问 时 间 data2， 只 包括 年 和 月 ; 网 站 的 IP 地 址 ip。 
要 获取 网 站 当月 IP 访问 量 , 其 查询 的 条 件 必须 从 时 间 data2 着 手 , 查询 出 当前 时 间 符合 data2 的 数 
。 代 码 如 下 : 


错 


<?php 
$query_4="select * from tb_count10 where data2=".substr(date("Y-m-d"),0,7).”"; 
$result_ 4=mysql_query($query_4); 


2 
Ls 技巧 应 用 substrO) 函 数 可 以 获取 当前 日 期 中 年 和 月 的 值 ， 将 获取 的 值 与 数据 表 中 data2 字段 的 
值 进行 比较 ， 从 中 读 取出 符合 条 件 的 数据 。 
然后 ， 将 获取 到 的 卫 地 址 写 入 到 一 个 数组 中 。 代 码 如 下 : 
while($myrow_4=mysql_fetch_array($result_4))}{ 
$counts_2[]=$myrow['ip]; /将 获取 的 ip 的 值 赋 给 变量 
} 


最 后 ， 先 应 用 array_ unique() 函 数 ， 去 除数 组 中 重复 的 值 ， 再 统计 出 数组 中 元 素 的 数量 ， 
即 总 的 名 访 问 量 。 


Sresults_4=array_unique($counts_4); // 去 除数 组 中 重复 的 值 
$countes_4=count($results_4); /获取 数组 中 值 的 数量 ， 即 总 的 IP 访问 量 


?> 


2.7.3 将 数组 中 的 数据 写 入 到 字符 串 中 


将 数组 中 的 数据 写 入 到 字符 串 中 ， 主 要 应 用 的 是 current() 函 数 和 implode() 函 数 。 
(1) current() 函 数 
该 函数 返回 数组 中 的 当前 单元 。 每 个 数组 中 都 有 一 个 内 部 的 指针 ， 初 始 指向 插入 到 数组 中 的 第 一 


@ 
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个 单元 。 该 函数 返回 当前 被 内 部 指针 指向 的 数组 单元 的 值 ， 并 不 移动 指针 。 如 果 内 部 指针 指向 超出 了 
数组 的 最 大 单元 ， 将 返回 false。 语 法 如 下 : 


mixed current( array array) 


(2) implode0) 函 数 
该 函数 将 数组 的 内 容 组 合成 一 个 字符 串 。 语 法 如 下 : 


string implode(string glue, array pieces) 


参数 glue 是 字符 之 间 的 分 隔 符号 。 
这 里 实际 是 将 指定 月 份 每 天 的 数据 写 入 到 字符 串 中 ， 实 现 对 网 站 指定 月 份 的 访问 量 进行 分 析 。 要 
完成 该 操作 ， 首 先 要 以 指定 的 月 份 为 查询 条 件 ， 查 询 出 指定 月 份 的 网 站 访问 量 的 数据 。 代 码 如 下 : 
<?php 
$query="select counts,data1 from tb_count10 where data2=".$_POST[select]." order by data1 "; 
$result=mysql_query($query); 


然后 通过 while 语句 ， 循 环 读 取 其 中 的 数据 ， 并 且 应 用 current(O) 函 数 将 数组 中 当前 的 元 素 写 入 到 数 
组 $results 中 。 

最 后 应 用 implode(0) 函 数 ， 将 数组 $results 中 的 数据 写 入 到 一 个 字符 串 变 量 $Slmbs 中 ， 以 “,” 为 分 隔 
符 ， 完 成 将 指定 月 份 每 天 的 数据 写 入 到 字符 串 中 的 操作 。 代 码 如 下 : 


while($myrow=mysql_fetch_array($result)X{ 
$results0=$myrow['counts']; 
$lImbs=implode(",",$results); 

} 


2.7.4 动态 生成 柱 形 图 分 析 网 站 访问 量 


动态 生成 柱 形 图 分 析 网 站 访问 量 ， 主 要 应 用 Jpgraph 类 库 根据 从 数据 库 中 读 取 的 数据 生成 柱 形 图 ， 
完成 对 网 站 访问 量 的 分 析 。 其 前 提 是 必须 在 项 目 中 载 入 Jpgraph 类 库 ， 由 于 其 属于 第 三 方 的 组 件 ， 在 本 
书 的 光盘 源码 中 没有 提供 ， 所 以 需要 读者 自行 下 载 ， 并 且 将 下 载 的 sr 文件 夹 复制 到 项 目 mA02\05 文 
件 夹 下 ， 然 后 才 可 以 完成 动态 生成 柱 形 图 分 析 网 站 访问 量 的 操作 。 


2.8 本 章 小 结 


本 章 通 过 网 页 计数 器 带领 读者 详细 讲解 一 个 计数 器 的 使 用 流程 。 同 时 ， 在 网 页 计数 器 的 使 用 过 程 
中 ,采用 了 Session 技术 、Cookie 技术 ， 使 整个 页 面 设计 思路 更 加 清晰 。 通 过 本 章 的 学 习 ， 读 者 不 仅 可 
以 了 解 一 般 网 站 的 访问 量 的 统计 ， 而 且 可 以 更 加 熟悉 Session、Cookie 的 使 用 。 


AS < 
性 
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文件 的 上 传 与 下 载 是 一 个 完整 的 网 站 应 该 具有 的 功能 之 一 ， 如 果 
网 站 中 不 具备 这 样 的 功能 ， 给 浏览 者 的 感觉 是 这 个 网 站 的 开发 不 是 很 
完整 。 本 章 将 全 面 介绍 PHP 的 上 传 与 下 载 技术 ,从 php.ini 文件 的 配 
置 ， 到 上 传 函 数 的 使 用 ， 到 最 后 的 动手 实践 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

由 文件 目录 及 文件 的 相关 操作 

由 文件 的 上 传 

让 文件 的 下 载 

让 文件 重 名 自动 检测 

M 利用 下 载 码 实现 文件 搜索 

由 如何 限 制 用 户 的 空间 使 用 大 小 

让 文件 上 传 的 相关 配置 
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3.1 上 传 与 下 载 模块 概述 


3.1.1 模块 概述 


使 用 过 ASP 的 读者 一 定 会 对 ASP 上 传记 忆 犹 新。 要 在 ASP 中 实现 上 传 ， 或 者 使 用 Active 组 件 上 
人 5 开 省 入“ 太 站 人 位 壬 筑 大 组 件 上 人 这 对 于 新 于 术 膏 和 训 理 出 生 民 7 鞭 闻 对 于 成 生 如 各 完 的 
难度 。PHP 在 这 方面 要 比 ASP 方便 得 多 ， 一 个 上 传 函数 就 能 够 完全 实现 。 如 果 是 大 文件 上 传 或 下 
载 ， wa wa 在 本 模块 中 将 对 PHP 的 上 传 和 下 载 功能 的 实现 
进行 系统 全 面 的 讲解 。 


3.1.2 ”功能 结构 


为 了 读者 更 好 地 学 习 ， 下 面 给 出 上 传 下 载 模块 的 功能 结构 流程 图 。 

用 户 在 上 传 下 载 模块 中 通过 注册 为 本 模块 用 户 ， 然 后 登录 模块 进行 相关 操作 。 用 户 正 确 登 录 后 ， 
系统 将 自动 判别 访客 的 级 别 ， 然 后 根据 不 同 的 级 别 分 配 不 同 的 权限 。 用 户 未 登录 时 ， 身 份 为 游客 ， 可 
通过 下 载 码 下 载 未 公开 文件 。 用 户 成 功 登 录 后 ， 根 据 级 别 的 不 同 ， 可 以 执行 上 传 新 文件 、 删 除 已 上 传 
个 人 文件 和 更 改 文件 公布 状态 等 操作 。 

上 传 与 下 载 模块 的 功能 结构 如 图 3.1 所 示 。 


上 传 与 下 载 模块 


[ 
[_ 登录 用 户 | 未 登录 


普通 用 户 会 员 用 户 高 级 用 户 


图 3.1 上 传 下 载 模块 功能 结构 图 


人 


乔装 机 上 
| 
弄 闪 天 害 
弄 冲 网 秦 
| 
弄 冲 游 洪 


| 


3.1.3 程序 预览 


上 传 与 下 载 模块 由 多 个 页 面 组 成 ， 下 面 列 出 几 个 典型 的 页 面 ， 其 他 的 请 参看 光盘 中 的 程序 。 


@ 
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上 传 与 下 载 模块 的 登录 页 面 如 图 3.2 所 示 ,， 用 户 想 下 载 自己 上 传 的 文件 ， 则 首先 执行 登录 操作 ， 登 
录 成 功 后 才 可 以 在 个 人 信息 显示 处 单 击 “ 下 载 文件 ” 超 链接 进入 下 载 页 。 
ET 可 


上 传 文件 | 下 载 文件 


图 3.2 上传 下 载 模块 的 登录 页 面 
上 传 与 下 载 模块 的 主页 如 图 3.3 所 示 。 


MR 区 网 络 v 生 
/ww mrbecdcom 国 首 页 | 四 nA 大 | 总 设 为 首页 


rs 民明 日 网 络 u 盘 
圣 证 码 ; bd 


ee WE :| rm 


袜 件 名 文件 类 型 。。 下 载 码 上 人 时 间 
找 之 让 痉 / 有 入 口 了 .0 区 装 包 .rar 音乐 4lofuezybeczeas 2011-02-25 08:53 28 


明日 科 蕉 编程 词典 


TY OBESS 
[ 综 雪 解难 


3.3 上传 下 载 模块 的 主页 
上 传 文件 页 面 如 图 3.4 所 示 ， 该 页 面 主要 是 上 传 新 文件 到 系统 分 配 的 空间 内 。 
国 上 传 文件 


上 传 文件 限制 : 100M 
Ci\Docunents and Settine MW [TAH lid 


Ex 


3.4 上传 文件 页 面 
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下 载 页 面 如 图 3.5 所 示 , 顾名思义 ， 该 页 面 主 要 是 展示 已 上 传 文件 的 相关 信息 ,用 户 可 以 通过 单 击 
该 页 面 上 文件 名 称 的 超 链接 下 载 该 文件 。 


TD: EE ER | 
文件 名 文件 类 型 下 载 码 上 传 时 间 更 疏 状 态 
请 IPB. 0 安装 包 . rar 音乐 ”4lbfae2bbac2eaa 。 2011-02-25 08:53:26 公开 


全 法 反选 莽 队 选择 首页 上 一 页 下 一 页 尾 页 当前 是 第 ! 页 / 共 1 页! 条 记录 [1 加 | 页 


图 3.5 下 载 页 面 
当前 用 户 为 游客 身份 时 ， 也 可 以 通过 文件 的 下 载 码 下 载 对 应 的 上 传 文件 ， 如 图 3.6 所 示 。 


"By: [ibfae2bbaczcad DN a#: [+9 El 


文件 名 文件 类 型 下 载 码 上 传 时 间 
IE8.0 安 装 包 . rar 音乐 4lbfae2bbac2casa 2011-02-25 08:53:26 


图 3.6 通过 下 载 码 下 载 文 件 
3.2 ”数据 库 设计 


本 节 通 过 实现 一 个 上 传 与 下 载 模块 ， 学 习 上 传 与 下 载 技术 在 实际 中 的 应 用 。 首 先 来 看 数据 库 的 设 
计 和 连接 。 


3.2.1 数据 库 分 析 


首先 定义 上 传 与 下 载 模块 使 用 MySQL 数据 库 ， 名 称 是 db_netu， 其 中 包含 3 个 表 ， 分 别 是 用 户 列 
表 、 上 传 文件 列表 和 文件 类 型 列表 。 其 中 用 户 列表 存储 用 户 的 相关 信息 ; 上传 文 件 列表 是 整个 上 传 下 
载 模块 的 核心 数据 表 ， 记 录 了 上 传 下 载 文件 的 相关 信息 ; 文件 类 型 列表 主要 记录 上 传 文件 的 种 类 。 数 
据 库 db_netu 包含 的 数据 表 如 图 3.7 所 示 。 
中 服务 器 : Ilocalhost ， 成 数据 库 - db netm 
加 结构 “有 SQL 万 搜索 品 查 询 。 总 导出 “本 Import ， 侨 操 作 


表 类 型 整理 说 明 
tb_member MyIsAM gb2312_chinese_ci 用 户 列表 
tb_upfile 。 MySAM gb2312_chinese_ci 上 传 文件 列表 
tb_uptype 。 MyIsAM gb2312_chinese_ci 文件 类 型 列表 


3.7 数据 库 db_netu 包含 的 数据 表 


第 3 章 上 传 与 下 载 模块 (PHP+Ajax+MySQL 实现 ) 


3.2.2 ”创建 数据 表 


1. tb_member 表 
tb_member 表 记 录 注 册 用 户 信息 ， 包 括 文件 名 称 、 密 码 保护 、 密 保 问 题 、E-mail、 用 户 权限 以 及 空 
间 使 用 情况 等 信息 。tb_member 表 结构 及 说 明 如 图 3.8 所 示 。 
2. tb_upfile 表 
tb_upfile 表 记 录用 户 上 传 的 文件 信息 ， 包 括 文件 名 称 、 文 件 类 型 、 下 载 码 等 信息 。tb_upfile 表 结构 
及 说 明 如 图 3.9 所 示 。 
加 服务 器 : localhost 加 数据 库 : db_netu ， 国 表 :tb_member 加 服务 器 : localhost ， 轧 数据 库 : db_netu ， 国 表 :tb_upfile 
_ 叶 浏览 果 结 构 ,好 SQL 万 搜索 丝 新 入 此 导出 并 Import 竹 换 作 _ 国 浏览 。 临 结 构 “ 丸 sQL 户 扫 索 3 插入 虹 导 出 ”并 Import 笑 操 


字 有 & 类 型 整理 是 Null 正 逢 字段 类 型 整理 说 明 
性 id int(4) 自动 编号 
本 | it) 百 【| BR fllename varchar(200) gb2312-chinese-ci 文件 名 称 
厂 name varchar(50) 。 gb2312_chinese_ci 要 文件 名 称 filepath varchar(200) gb2312_chinese_cl 。 “保存 路 径 
厂 “password varchar(50) gb2312_chinese_ci 理 用 户 密码 filetype varchar(100) gb2312_chinese_ci 文件 类 型 
厂 question varchar(50) gb2312_chinese_cl 。 否 。 密码 保护 upauthor varchar(200) gb2312_chinese_cl 。 上 使 用 户 
厂 answer varchar(50) gb2312_chinese_ci 理 密 保 问题 uprime tmesramp 上 传 时 间 
厂 email varchar(200) gb2312_chinese_cl 。 否 Email chkdownnum varchar(20) ”gb2312_chinese-cl 。 下 载 码 
厂 rt int() 要 用 户 权限 ispub int0) 是 否 公开 
厂 cantem bigintso) 否 。 守 间 使 用 情况 fesize blgintc20) 六 村 个 
图 3.8 tb_member 表 结构 及 说 明 图 3.9 tb_upfile 表 结构 及 说 明 
3. tb_uptype 表 


tb_uptype 表 规 定 上 传 文件 的 类 型 ， 用 户 可 以 将 文件 上 传 到 不 同 的 类 型 中 ， 以 方便 日 后 查找 。 
tb_uptype 表 结 构 及 说 明 如 图 3.10 所 示 。 
中 服务 器 : localhost ， 加 数据 库 : db_netu 。 ， 贺 表 : tb_uptype 
_ 园 测 览 ” 阿 结 构 “ 录 sQL 万 搜索 也 插入 障 导 出 ” 障 Import 稀 操 作 


字段 类 型 整理 额外 说 明 
过 int(®) auto_increment 自动 编号 
genrename varchar(200) gb2312_chinese_ci 类 型 名 称 


图 3.10 tb_uptype 表 结 构 及 说 明 


3.2.3 ”数据库 连接 文件 


PHP 连接 数据 库 并 非 难事 ， 但 是 在 每 次 应 用 数据 库 连 接 时 都 重复 输入 类 似 代 码 就 会 耽误 大 量 的 时 
间 ， 同 时 也 降低 了 代码 的 书写 效率 和 可 读 性 。 所 以 ， 本 模块 中 将 数据 库 的 连接 方法 封装 到 一 个 类 中 ， 
这 样 ， 在 应 用 数据 库 连 接 操 作 时 ， 调 用 该 类 文件 即 可 。conn\conn.php 文件 的 关键 代码 如 下 : 


<?php 
class opmysql{ 
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private $host = 'localhost’; 
private $name = root'; 
private $pwd = '111"; 
private $dBase = 'db_netu'; 
private $conn = "; 
private $result = "; 
private $msg = "; 

private $fields; 

private $fieldsNum = 0; 

private $rowsNum = 0; 

private $rowsRst = "; 

private $filesArray = array(); 

private $rowsArray = array(); 

// 初 始 化 类 

function __construct($host=",$name=",$pwd=", 
if($host!= ") 

S$this->host = $host; 

if($name != ") 

$this->name = $name; 

if($pwd {= ") 

$this->pwd = $pwd; 

if($dBase !=") 

S$this->dBase = $dBase; 

S$this->init_conn(); 


外 
// 链 接 数据 库 
function init_conn(){ 


/服务 器 地 址 
/登录 账号 

/登录 密码 
/数据 库 名称 
/数据 库 连 接 资源 
/结果 集 
/返回 结 果 

// 返 回 字段 

/返回 字段 数 

// 返 回 结果 数 

// 返 回 单条 记录 的 字段 数组 
/返回 字 段 数组 

// 返 回 结果 数组 


$dBase='"){ 


S$this->conn=@mysql_connect($this->host,$this->name, $this->pwd); 


@mysql_select_db($this->dBase, $this->conn); 
mysql_query("set names gb2312"); 

} private $dBase = 'db_netu'; 

// 省 略 部 分 代码 


/| 数据 库 名 称 


另外 ， 该 自 定义 类 中 还 包含 了 一 些 常 用 的 数据 库 操作 方法 ， 具 体 方法 和 相关 说 明 请 查看 光盘 中 的 


源 程序 ， 这 里 不 再 袭 述 。 


3.3 用 户 注 册 


3.3.1 注册 模块 概述 


上 传 与 下 载 模块 是 一 个 面向 公开 用 户 的 个 人 网 络 存储 空间 程序 ， 


自助 式 的 用 户 注册 是 其 功能 模块 之 


一 ， 新 用 户 可 以 通过 单 击 登录 模块 中 “注册 ” 超 链接 来 进入 自助 式 用 户 申 请 页 面 。 其 运行 结果 如 图 3.11 


所 示 。 


@ 
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电子 相册 注册 页 面 
共 喜 您 ， 该 用 户 名 可 以 使 用 ! 
密码 符合 要 求 。 窗 码 强度 : 弱 
确认 密码 : eeeeee 密码 输入 正确 
密 保 问题 : |111 
密 保 答案 : |111 
Email : fningrosft@ningrisoft. con | 
注册 ”详细 信息 


图 3.11 用 户 注册 运行 结果 
3.3.2 ”注册 页 面 设计 
注册 页 面 的 设计 没有 太 多 实质 性 的 理论 可 言 ， 其 主要 是 通过 表单 来 获取 用 户 注 册 的 相关 信息 ， 将 


数据 提交 到 register_chk.php 文件 中 ， 完 成 用 户 注册 信息 的 处 理 。 注 册页 面 文件 register.php 的 相关 代码 
如 下 : 


<?php 

session_start(); 

header('Content-Type:text/html;charset=gb2312"); 

?> 

<table id="regfm" width="500" border="0" cellspacing="0" cellpadding="0" align="center"> 
<tr> 

<td height="25" colspan="3" align="center" valign="middle"> 电 子 相册 注册 页 面 </td> 
</tr> 

<tr> 

<td width="100" height="25" align="right" valign="middle"> 注 册 名 称 : </td> 

<td width="200" height="25" align="left" valign="middle">&nbsp; 

<input id="regname" name="regname" type="text" class="txt" /></td> 

<td height="25"><div id="namediv" class="regdiv"> 名 称 由 英文 字母 和 数字 及 下 划 线 组 成 </div></td> 
</tr> 

<tr> 

<td width="100" height="25" align="right" valign="middle"> 注 册 密码 : </td> 

<td width="200" height="25" align="left" valign="middle">&nbsp; 

<input id="regpwd1" name="regpwd1" type="password" class="txt" /></td> 

<td height="25"><div id="pwddiv1" class="regdiv"> 请 输入 密码 </div></td> 

</tr> 

<tr> 

<td width="100" height="25" align="right" valign="middle"> 确 认 密 码 : </td> 

<td width="200" height="25" align="left" valign="middle">&nbsp; 

<input id="regpwd2" name="regpwd2" type="password" class="txt" /></td> 

<td height="25"><div id="pwddiv2" class="regdiv"> 确 认 密 码 </div></td> 

</tr> 

</table> 

</table> 

<div id="morediv" style="display:non;"> 


<!-- 省 略 部 分 代码 --> 
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</table> 

</div><center> 

<div style=" text-align:center;text-align:center; width:50%; height:25px; line-height:25px;"> 
<button id="regbtn" class="btn" disabled="disabled"> 注 册 </button>&nbsp; 

<button id="morebtn" class="btn"> 详 细 信 息 </button></div></center> 


3.3.3 Ajax 无 刷新 技术 验证 注册 信息 


新 用 户 在 注册 新 账号 时 ， 如 果 在 每 一 次 提交 注册 信息 之 后 才 返 回 不 符合 要 求 的 错误 信息 ， 不 但 降 
低 了 执行 效率 ， 而 且 也 会 消磨 用 户 的 耐心 ， 这 样 一 来 ， 用 户 之 前 填写 的 所 有 注册 信息 会 随 着 操作 失败 
而 消失 。 为 了 让 用 户 及 时 得 到 反馈 信息 ， 注 册 模块 采用 目前 比较 流行 的 Ajax 技术 ， 对 注册 信息 执行 无 
刷新 验证 操作 。 这 里 以 验证 注册 用 户 名 为 例 ， 该 过 程 是 通过 js 文件 目录 中 的 registerjs 文件 来 实现 的 。 
关键 代码 如 下 : 


function $(idX{ 

return document.getElementByld(id); /获取 标签 ID 
} 

$(regname').onblur = function(X{ 

name = $(regname').value; 

if(cname1 == yes'){ 

xmlhttp.open(get,chkname.php?name='+name,true); 
xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readyState == 4){ 

if(xmlhttp.status == 200){ 

var msg = xmlhttp.responseText'; 

if(msg == '1){ // 判 断 用 户 名 是 否 被 占用 
$('namediv').innerHTML="<font color=green> 恭 喜 您 ， 该 用 户 名 可 以 使 用 !</font>"; 
cname2 = yes'; 

}else if(msg == '2"){ 

$(namediv).innerHTML="<font color=red> 用 户 名 被 占用 ! </font>"; 
cname2 = "; 

}else{ 

$('namediv').innerHTML="<font color=red>"+msg+"</font>"; 

cname2 ="; 

} 

> 

3 


} 
xmlhttp.send(null); 
chkreg(); 

3 

了 


在 registerjs 文件 中 ， 无 刷新 调用 chkname.php 文件 ， 对 注册 用 户 名 进行 验证 ， 根 据 chkname.php 
文件 的 返回 值 判断 该 用 户 名 是 否 可 用 。chkname.php 文件 的 关键 代码 如 下 : 


<?php 


@ 


第 3 章 上 传 与 下 载 模 块 (PHP+Ajax+MySQL 实现 ) 


session_start(); /开启 会 话 

include_once "conn/conn.php"; /1 加 载 数据 库 配置 文件 

S$reback = '0'; // 定 义 验证 返回 值 

$sql = "select * from tb_member where name=".$_GET[name'].”"; 

$num = $conne->getRowsNum($sql); /| 执行 数据 库 查询 操作 

if($num == 1X // 判 断 是 否 存 在 匹配 结果 

S$reback = '2'; // 如 果 存 在 ， 则 将 验证 返回 值 赋值 2 
}else if($num == OX 

S$reback = '1'; // 如 果 不 存在 ， 则 将 验证 返回 值 赋值 1 
}else{ 

$reback = $conne->msg_error(); // 输 出 错误 信息 

} 

echo $reback; // 输 出 验证 返回 值 

他 > 


3.3.4 注册 信息 处 理 


创建 register_chk.php， 获 取 注 册页 面 register.php 中 提交 的 注册 信息 ， 最 终 将 用 户 的 注册 信息 添加 
到 指定 的 数据 表 中 ， 同 时 为 新 注册 用 户 创建 一 个 属于 自己 的 存储 文件 目录 。 注 册 信 息 处 理 文件 
register_chk.php 的 相关 代码 如 下 : 


<?php 

session_start(); 

header('Content-Type:text/html;charset=gb2312'); 

include_once 'conm/conn.php'; 

$reback = '0'; 

$sql = "insert into tb_member(name,password,question,answer,email) 
values(".trim($_GET[name]).",".md5(trim($_GET[pwd])).",".$_GET[question]."， 
".$_GET[answer].",".$_GET[email].")"; 

$num = $conne->uidRst($sql); 

if($num == 1X{ 

$_SESSION[name']=$_GET[name]; 
$username=$_SESSION[name]; 

$dirname="../upfile/$usemame"; /创建 一 个 和 用 户 名 相同 的 文件 夹 
mkdir($dimame,0700); 

$reback = "1"; 

$file = 'index.txt'; 

$newfile="../upfile/$usemame/index.php"; 

if (lcopy($file, $newfile)) { 

echo "failed to copy $file..\n"; 

} 

}else{ 


$reback = $conne->msg_error(); 


和 
echo $reback; 
本 
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3.4 用 户 登 录 


3.4.1 登录 模块 概述 


上 传 与 下 载 模块 通过 用 户 登 录 实现 个 人 空间 划分 、 

控制 上 传 文件 的 公开 或 者 隐藏 ， 以 及 对 个 人 上 传 文件 的 MF, 人 

管理 。 用 户 登录 的 运行 结果 如 图 3.12 所 示 。 ， ee 
用 户 登 录 模块 中 包含 了 用 户 登 录 、 新 用 户 注册 、 找 


欢迎 光临 ; leonsk 


回 密码 等 功能 , 同时 输出 了 用 户 的 权限 、 剩余 空间 大 小 、 ps: ee 
剩余 空间 百分比 以 及 上 传 文件 和 下 载 文件 的 超 链接 。 由 2 | s em 
于 篇 幅 所 限 ， 下 面 只 对 用 户 登录 模块 中 的 一 些 特殊 功能 i sr et 
进行 详细 讲解 ， 其 完整 内 容 请 参考 本 书 光盘 中 的 源 代 i 


码 ， 这 里 不 再 袭 述 。 
3.4.2 用户 权 限 输出 以 及 空间 使 用 计算 

在 用 户 登 录 后 ， 将 输出 用 户 的 权限 以 及 用 户 个 人 空间 的 使 用 情况 。 下 面 就 介绍 如 何 判断 登录 用 户 
的 权限 ， 以 及 空间 使 用 情况 的 计算 。 


(1) 在 index.php 中 , 根据 Session 变量 传递 的 登录 用 户 名 称 , 从 数据 库 中 读 取 该 用 户 的 个 人 信息 
对 tb_member 数据 表 中 right 字段 的 值 进行 判断 ， 进 而 输出 用 户 的 权限 。 关 键 代 码 如 下 


图 3.12 登录 模块 运行 结果 


<?php 

S$rightsql = "select * from tb_member where name = ".$_SESSION[name'].""; // 查 询 语句 

S$rightarr = $conne->getRowsRst($rightsql); // 返 回 查询 结果 集 ， 判 断 用 户 权限 
$conne->close_rst(); 

if($rightarr[right]==2){ /如 果 返 回 查询 值 为 2， 判断 为 高 级 用 户 


echo "<span class=\"vip1\">[ 高 级 用 户 ]</span>"; 
}else if($rightarr[right]==1){ 


echo "<span class=\"vip2\">[ 会 员 用 户 ]</span>"; // 如 果 返 回 查询 值 为 1， 判断 为 会 员 用 户 
jelsef 

echo "<span class=\"vip3\">[ 普 通用 户 ]</span>"; /1 否则 判断 为 普通 会 员 

} 

> 


(2) 计算 用 户 的 空间 使 用 情况 。 首 先 ， 从 数据 库 中 读 取 权 限 字段 中 的 数值 ， 判 断 当前 用 户 的 登录 
权限 ， 通 过 用 户 的 权限 来 设 定 其 使 用 空间 大 小 ; 然后 ， 从 数据 库 中 读 取 该 用 户 的 空间 使 用 情况 ， 将 其 
分 配 空间 的 大 小 与 该 用 户 已 使 用 空间 大 小 做 差 值 ， 最 后 ， 将 取得 结果 进行 格式 化 输出 ， 显 示 当 前 登录 
用 户 的 剩余 空间 和 剩余 空间 的 百分比 。 相 关 代码 如 下 : 


<p> 您 剩余 空间 : 
<?php 


@. 
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if($rightarr[right]==2){ /如果 权限 为 高 级 用 户 ， 则 其 分 配 空间 为 SMB 
if(5000000-$rightarrf' content]<=1000) { // 判 断 用 户 空间 是 否 不 足 

echo "已 满 ! "; 

jelse{ 

$formatted=(5000000-$rightarrf'content])/1000000; // 对 剩余 空间 大 小 进行 换算 
S$forstr=sprintf("%-01.2f",$formatted); // 对 计算 的 空间 大 小 数值 进行 格式 化 

echo $forstr."MB"; 

echo "<p> 剩 余 率 :".sprintf("%-01.2f",((1-$rightarrfcontent]/5000000)*100))."%</p>"; 

// 对 用 户 的 剩余 空间 以 百分比 的 格式 化 输出 


jelse if(rightarr[right]==1){ 

if(3000000-$rightarrf'content]<=1000) { 

echo "已 满 !"; 

}else{ 

$formatted=(3000000-$rightarrf'content])/1000000; 
$forstr=sprintf("%-01.2f",$formatted); 

echo $forstr."MB"; 

echo "<p> 剩 余 率 :".sprintf("%-01.2f",((1-$rightarrrcontent]/3000000)*100))."%</p>"; 
} 

}else{ 

if(2000000-$rightarrf'content]<=1000){ 

echo "已 满 ! "; 

}else{ 

$formatted=(2000000-$rightarrf'content])/1000000; 
$forstr=sprintf("%-01.2f",$formatted); 

echo $forstr."MB"; 

echo "<p> 剩 余 率 :".sprintf("%-01.2f",((1-$rightarr['content]/2000000)*100))."%</p>"; 
} 

于 


?> 


3.4.3 为 用 户 划分 个 人 空间 文件 目录 


为 注册 的 每 一 个 用 户 创建 一 个 和 其 注册 名 同名 的 个 人 文件 目录 ， 这 样 不 但 可 以 方便 管理 员 对 个 人 
文件 的 管理 ， 同 时 也 避免 了 所 有 用 户 上 传 文件 到 同一 文件 目录 下 的 浆 端 。 

创建 个 人 文件 目录 的 操作 在 用 户 的 注册 处 理 页 register_chk.php 中 完成 。 首 先 获取 表单 提交 的 用 户 
注册 信息 ， 然 后 执行 添加 语句 将 注册 信息 添加 到 指定 的 数据 表 ， 最 后 通过 mkdir() 函 数 在 服务 器 指定 的 
目录 下 创建 一 个 与 用 户 名 相同 的 文件 夹 。 关 键 代码 如 下 : 


<?php 

session_start(); 

header('Content-Type:text/html;charset=gb2312'); 

include_once 'conn/conn.php'; 

S$reback = '0'; 

$sql = "insert into tb_member(name,password,question,answer,email) 
values(".trim($_GET[name]).",".md5(trim($_GET[pwd])).",".$_GET[question]."， 
".$_GET[answer].",".$_GET[email].")"; 

$num = $conne->uidRst($sq)); 

if($num == 1X{ 
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$_SESSION['name'] = $_GET[name'; 
$username=$_SESSION['name'; 


$dirname="../upfile/$usemame"; /设置 文件 夹 的 路 径 及 名 称 
mkdir($dimame,0700); /创建 一 个 和 用 户 名 相同 的 文件 夹 
$reback = '1"; 


3.4.4 ”防止 用 户 文件 目录 的 非法 访问 


某 些 用 户 可 能 通过 不 正当 方式 获取 其 他 用 户 文件 夹 的 地 址 ， 这 样 就 可 以 直接 通过 地 址 访问 其 他 用 
户 文件 夹 ， 获 得 相关 的 内 容 并 下 载 其 他 用 户 文件 。 为 了 避免 这 一 浆 端 ， 在 为 每 个 用 户 创建 自己 文件 夹 的 同 
时 ， 在 自己 的 文件 夹 下 创建 了 一 个 index.php 文件 ， 以 此 来 屏蔽 直接 通过 地 址 栏 访问 文件 夹 下 内 容 的 方式 。 

首先 ， 在 login 文件 夹 下 创建 一 个 名 为 index.txt 的 文本 文档 。 代 码 如 下 : 

<?php 

echo "请 使 用 正确 的 访问 路 径 !"; 

3 

href="../../index.php"> 这 里 </a> 返 回 // 单 击 返回 首页 

</p> 

然后 ， 在 用 户 注册 的 处 理 页 register_chk.php 中 ， 在 完成 用 户 注 册 信 息 的 添加 之 后 ， 将 index.txt 文 
本 文件 复制 到 新 创建 的 用 户 个 人 文件 夹 下 ， 并 且 命名 为 index.php。 关 键 代码 如 下 : 


Sfile = 'index.txt'; 

$newfile="../upfile/$username/index.php"; /设置 复 制 新 路 径 名 称 

if (lcopy($file, $newfile)) { // 将 index.txt 复制 到 新 路 径 
echo "failed to copy S$file..\n"; 

} 

}else{ 


$reback = $conne->msg_error(); 


} 
echo $reback; 
ns 


3.5 找 回 密码 


3.5.1 找 回 密码 概述 


长 时 间 未 登录 的 用 户 可 能 会 忘记 网 站 的 登录 密码 ; 另外 ， 一 些 用 户 的 密码 也 可 能 会 被 他 人 非法 更 
改 。 所 以 ， 找 回 密码 模块 即 为 个 人 用 户 取 回 密码 提供 了 一 个 有 效 平台 ， 这 样 就 保证 每 一 位 用 户 能 够 及 
时 找 回 登录 密码 。 找 回 密码 的 运行 结果 如 图 3.13 所 示 。 

在 本 模块 中 ， 找 回 密码 采用 的 是 密 保 问题 与 密 保 答案 比较 的 方式 ， 首 先 用 户 输入 注册 的 用 户 名 ， 
然后 输入 密 保 问题 和 密 保 答 案 ， 最 后 对 用 户 的 提交 问题 和 答案 与 数据 库 中 存储 的 数据 进行 比较 ， 如 果 
问题 和 答案 都 正确 ， 则 该 用 户 可 以 执行 密码 的 修改 操作 ， 重 新 设置 新 的 密码 。 


@_ 
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http//127.90.UTM/O3logr/found.php Http//127.00.VTM/03/iogin/found.php 


四 Intemet | 保护 模式 : 禁 组 ” 所 100% > 人 @ internet | 保护 模 式 禁 入 ” 太 100% ~ 


图 3.13 ” 找 回 密码 的 运行 结果 


3.5.2 ” 找 回 密码 页 面 设计 
在 找 回 密码 页 面 中 ， 关 键 是 创建 表单 ， 提 交 注 册 的 用 户 名 、 密 保 问题 和 密 保 答案 。 同 时 还 嵌入 了 


两 个 不 同 ID 值 的 div 层 ， 通 过 控制 div 层 样式 的 隐藏 和 展开 来 实现 找 回 密码 表单 和 更 改 密码 表单 之 间 


的 切换 操作 。 找 回 密码 文件 found.php 的 关键 代码 如 下 : 


<!- 找 回 密码 部 分 -> 

<div id="foundinfo"> 

<table width="250" border="1" align="center cellpadding="0" cellspacing="0"> 
<tr> 

<td height="25" colspan="2" align="center" valign="middle"> 密 码 找 回 </td> 
</tr> 

<tr> 

<td width="100" height="25" align="right" valign="middle"> 找 回 账号 : </td> 
<td height="25">&nbsp; 


<input id="foundname" type="text" style=" width: 100px; height:15px; border:1px #000000 solid;" /></td> 


</tr> 

<!-- 省 略 部 分 代码 --> 

<tr> 

<td height="25" colspan="2" align="center" valign="middle">&nbsp; 
<button id="step1" class="btn"> 下 一 步 </button></td> 

</tr> 

</table> 

</div> 

<!-- 更 改 密码 部 分 -> 

<div id="changepwd" style="display:none;"> 

<table width="250" border="1" align="center" cellpadding="0" cellspacing="0"> 
<tr> 

<td height="25" colspan="2" align="center" valign="middle"> 更 改 密码 </td> 
</tr> 

<tr> 

<td width="100" height="25" align="right" valign="middle"> 输 入 密码 : </td> 
<td height="25">&nbsp; 


<input id="pwd1" type="password" style=" width: 100px; height:15px; border:1px #000000 solid;" /></td> 


</tr> 
<tr> 
<td width="100" height="25" align="right" valign="middle"> 二 次 密码 : </td> 
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<td height="25">&nbsp; 

<input id="pwd2" type="password" style=" width: 100px; height:15px; border1px #000000 solid;" /></td> 
</tr> 

<tr> 

<td height="25" colspan="2" align="center" valign="middle">&nbsp; 

<button id="step2" class="btn"> 完 成 </button></td> 

</tr> 

</table> 

</div> 


在 found.php 文件 中 , 载 入 了 js\found.js 脚本 文件 , 通过 Ajax 无 刷新 技术 调用 found_chk.php 文件 ， 
对 用 户 提交 的 密 保 问题 和 密 保 答案 进行 验证 ， 并 且 根 据 found_chk.php 文件 中 的 返回 值 控制 div 层 样式 
的 隐藏 和 显示 。foundjs 脚本 文件 的 代码 如 下 : 


S$('step1').onclick = function(){ 

if($(foundname').value := " && $('question').value != " && $(answer).value != "){ 
xmlhttp.open(get',found_chk.php?act=sel&foundname='+$(foundname') 
.value+'&question='+$(question').value+'&answer='+$(answer).value,true); /验证 找 回 密 码 预 设 信息 
xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readystate == 4 && xmlhttp.status == 200X{ 

msg = xmlhttp.responseText; 


if(msg == "1){ 
$('foundinfo').style.display = none'; /更 改 display 属性 隐藏 foundinfo 层 
$(changepwd').style.display = "; /更 改 display 属性 显示 changepwd 层 
}else{ 

alert(' 填 写 信息 错误 ! 小 

} 

} 

} 

xmlhttp.send(null); 

} 

} 


3.5.3” 找 回 密码 处 理 


创建 找 回 密码 处 理 文件 found_chk.php。 该 处 理 文件 主要 包括 两 个 不 同 的 处 理 方法 ， 即 验证 密 保 问 
题 和 更 改 密码 。 首 先 通过 URL 获取 act 的 参数 值 来 判断 用 户 执 行 的 操作 ， 然 后 根据 选择 的 act 参数 来 
执行 相应 的 方法 ， 对 用 户 提交 的 密 保 问 题 和 密 保 答案 进行 验证 ， 或 者 完成 对 用 户 提交 新 密码 的 更 新 操 
作 。 关 键 代码 如 下 : 


<?php 
header('Content-Type:text/html;charset=gb2312"); 
include_once 'conn/conn.php'; 

$act = $_GET['act]; 


S$reback = '0'; 

if($act == 'sel'}{ // 判 断 用 户 选择 的 操作 
$name = $_GET[foundname']; /获取 找 回 密码 用 户 名 
$question = $_GET['question'; // 找 回 密码 问题 


@© 
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$answer = $_GET['answer']; // 找 回 密码 

$sql="select * from tb_member where name=". $name." and question=".$question." and answer=".$answer.""; 
$num = $conne->getRowsNum($sql); 

}else if($act == 'up'){ 

$name = $_GET[foundname']; 

$pwd = $_GET[pwd]; 

$sql = "update tb_member set password = ".md5($pwd)." where name = ".$name.""; 


$num = $conne->uidRst($sql); /| 执行 数据 库 更 新 操作 
if($num = ") 

S$reback = "1"; 1/ 返回 值 为 1 

}else{ 

$reback = $sql; 

} 

echo $reback; 

J 


a 
has 


3.6 文件 上 传 


3.6.1 文件 上 传 概述 


作为 网 络 移动 存储 工具 ， 上 传 是 必 不 可 少 的 功能 之 一 。 用 户 可 以 通过 登录 个 人 账户 ， 单 击 “ 上 传 
文件 ” 超 链接 ， 进 入 文件 上 传 页， 将 自己 想 要 上 传 的 工具 、 文 档 、 音 频 或 视频 等 多 种 类 型 的 文件 上 传 
到 个 人 空间 中 。 文 件 上 传 页 面 的 运行 结果 如 图 3.14 所 示 。 
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3.14 文件 上 传 页 面 的 运行 结果 
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3.6.2 ”文件 上 传 表单 设计 


文件 上 传 要 通过 表单 中 的 文件 域 来 提交 , 提交 到 处 理 页 中 通过 $_FILES[] 全 局 变量 和 move_upload_ 
file0) 函 数 完成 文件 的 上 传 操 作 。 

上 传 文件 的 表单 存储 于 upfile.php 文件 中 .首先 ,创建 form 标记 ,指定 POST 方法 提交 ,设置 enctype 
属性 等 于 mnultipart/form-data， 设 置 处 理 页 为 upfile_chk.php。 然 后 ， 创 建文 件 域 ， 选 择 上 传 文件 。 接 着 ， 
创建 下 拉 列 表 框 展示 文件 类 型 ， 将 从 数据 库 读 取 的 文件 类 型 作为 下 拉 列 表 框 的 值 ， 创 建 下 拉 列 表 框 设 
置 文件 权限 。 最 后 ， 创 建 提交 按钮 ， 通 过 onclick 事件 调用 JavaScript 脚本 中 的 方法 验证 提交 数据 ， 以 
及 完成 提交 操作 。upfile.php 文件 的 关键 代码 如 下 : 


<!-- 上 传 表单 ， 处 理 页 为 upfile_chk.php--> 

<form method="post" enctype="multipart/form-data" action="?act=upfile"> 

<! 一 列表 项 --> 

<ul id="change"> 

<!-- 上 传 表单 ， 注 意 ，name 值 为 一 个 数组 -> 

<liid="addordel"> 

<input id="upname[]" name="upnamel" type="file" style=" width: 200px; background-color:#f0f0f0; " />&nbsp; 
<!-- 文 件 类 别 下 拉 列 表 ，name 也 是 数组 --> 

<select id="foundtype[]" name="foundtype[ "> 


<?php 
$typesql = "select genrename from tb_uptype"; /查询 语句 

$typearr = $conne->getRowsArray($typesql); /返回 查询 结果 集 
$conne->close_rst(); /清空 数组 
foreach($typearr as $valueX{ // 循 环 输出 类 别 列表 
?> 
<option value="<?php echo $value['genrename']; ?>"><?php echo $value['genrename']; ?></option> 
<?php 
} 
?> 
</select>&nbsp; 


<!-- 判 断 文件 是 否 公 开 。0 表示 不 公开 ; 1 表示 公开 --> 
<select id="ispub[]" name="ispub[ "> 

<option value="0" selected="selected"> 不 公开 </option> 
<option value="1"> 公 开 </option> 

</select>&nbsp; 

<!-- 删 除 上 传 域 和 添加 上 传 域 --> 

</li> 

</ul> 

<input id="upbtn" name="upbtn" type="submit" value=" 上 传 " 
onClick="return chkfile()" style=" background-color:#fOf0f0; border: 1px #CCCCCC solid;" /> 
</form> 


3.6.3 ”文件 上 传 处 理 


创建 文件 上 传 处 理 页 upfile_ chk.php。 首 先 获取 表单 提交 的 数据 ， 定 义 文件 的 存储 路 径 。 然 后 判断 


@. 
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是 否 存在 重 名 文件 ， 判 断 上 传 文件 大 小 是 否 符合 要 求 ， 生 成 下 载 码 。 接 着 完成 文件 的 上 传 操作 ， 并 且 
将 文件 信息 、 下 载 码 和 空间 使 用 信息 存储 到 数据 库 中 。 最 后 输出 下 载 码 ， 返 回 上 一 页 。 关 键 代 码 如 下 : 


include_once 'conn/conn.php'; 
$filename = $_FILES[upname'][rname]; 


// 读 取 上 传 文件 名 ， 并 存 为 数组 


S$filetype = $_POST[foundtype']; // 读 取 上 传 文件 的 类 别 

S$tmpname = $_FILES[upname'][tmp_name']; // 读 取 临 时 文件 名 ,并存 为 数组 
S$tmpsize = $_FILES[upname'['size'; /上 传 文件 大 小 

$tmppub = $_POST[ispub']; /是 否 公 开 
$dirname=$_SESSION[name']; /| 选取 和 用 户 登录 名 相同 的 个 人 文件 夹 
$file_path = "upfile/$dimame/"; /文件 保存 路 径 

$max = 0; /文件 大 小 ， 初 始 化 为 0 
for($j=0;$j<count($_FILES[upname'][name']);$j++){ /| 判断 上 传 的 文件 是 否 已 存在 
$filearr=$_FILES[upname']['name’[$)]; /| 获取 上 传 的 文件 名 
$filenamei=$file_path. $filearr; /设置 上 传 文件 路 径 
if(file_exists($filenamei) 1/ 判断 同 目录 下 文件 是 否 存在 
$construct=0; 

else 

$construct=1; // 如 果 存 在 则 返回 0， 不 存在 则 返回 1 


} 
if($construct!=1){ 


// 判 断 文件 存在 ， 则 提示 文件 名 称 冲突 


echo '<script>alert(" 文 件 名称 冲 突 ! 请 重新 命名 后 上 传 !"); 


location=".$_SERVER['HTTP_REFERER'.";</script>'; 


和 

if(is_array($tmpsize)){ 
foreach($tmpsize as $value){ 

$max += $value; 

} 

if($max > 100000000 or $max <= 0){ 


// 如 果 变 量 $tmpsize 不 是 数组 
// 获 取 文件 总 的 大 小 


echo '<script>alert(" 上 传 文件 总 大 小 大 于 100MB， 请 重新 选择 "); 


location=".$_SERVER['HTTP_REFERE R]. ";</script>"; 


于 
jelsef 


/重新 选择 上 传 文件 


echo '<script>alert(" 上 传 文件 错误 ， 请 重新 检查 程序 "); 
location=".$_SERVER['HTTP_REFERER]…";</script>'; 


} 

$rightsql = "select * from tb_member where name = ".$_SESSION[name'].”"; 
S$rightarr = $conne->getRowsRst($rightsql); 
$conne->close_rst(); 
Switch($rightarrfright]){ 

case '0': 

$conmax=2000000; 

break; 

case 1': 

$conmax=3000000; 

break; 

Case'2": 

$conmax=5000000; 

} 


// 判 断 用 户 权 限 
// 按 当前 登录 用 户 的 权限 分 配 的 最 大 使 用 空间 
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echo "<p>" 
$lesarr=$conmax-$rightarr[' content]; // 将 该 用 户 剩余 空间 大 小 赋值 给 lesarr 变量 
if($conmax-$rightarr[content] <= 1000000) { // 如 果 个 人 剩余 空间 已 经 不 足 1MB 时 


echo '<script>alert(" 您 的 上 传 下载 空 间 不 足 ， 请 清理 后 上 传 文件 "); 
location=".$_SERVER['HTTP_REFERER'1].";</script>'， // 提 示 空 间 不 足 


3 
else if($lesarr-$max<=1000000) /| 如 果 上 传 文件 大 小 超过 剩余 空间 大 小 


. 

echo '<script>alert(" 您 的 上 传 下 载 空间 不 足 ， 请 整理 后 上 传 文件 "); 
location=".$_SERVER[IHTTP_REFERER].";</script>'; // 提 示 空 间 不 足 
} 

$chkdownnum = "; 

for($i = 0; $i < 15; $i++X{ 

$chkdownnum .= dechex(rand(0,15)); 


} 

for ($i = 0; $i < count($filename); $i++X{ // 通 过 循环 将 多 个 文件 全 部 上 传 
move_uploaded file($tmpname[$i,$file 一 path.$filename[$i]); 

// 添 加 数据 库 


$insertsql = 'insert into tb_upfile (filename,filepath,filetype,upauthor,chkdownnum,ispub,filesize) 
values(".trim($filename[$i]).",".S$file_path. $filename[$i].",". $filetype[$i].", 
".$_SESSION[name’].",".tim($chkdownnum).",.$tmppub[S$i].",.$tmpsize[$i].")'; 
$conne->uidRst($insertsql); 


} 
$maxtotal=$rightarr['content]+$max; /从 数据 库 得 到 该 用 户 已 使 用 空间 大 小 与 新 上 传 的 总 数 
// 更 新 数据 库 


$csql="update tb_member set content = $maxtotal where name= ".$_SESSION[name].”"; 
$conne->uidRst($csql); 

echo '<div style=" padding-top: 30px;"> 文 件 上 传 成 功 ， 可 以 通过 下 载 码 : 

<font color="#FF0000">'.$chkdownnum.'</font> &nbsp; 对 文件 进行 下 载 。</div>'; 

echo "<p> 点 <a href=\"index.php?act=up\"><span style=\"color:red\"> 这 里 </span></a> 继 续 上 传 其 他 文件 ! </p>"; 


A 
说明 下 载 码 是 一 组 16 位 长 的 十 六 进 制 字符 囊 . 先 由 randO 函 数 生成 一 个 0~15 之 间 的 随机 数 ， 
然后 使 用 dechex0 〇 函数 将 该 数 转 为 十 六 进 制 表示 ， 最 后 使 用 for 循环 重复 16 次 。 如 果 一 次 上 传 多 个 
文件 ， 那 么 所 有 文件 的 注册 码 为 同一 个 。 


3.7 文件 下 载 


3.7.1 功能 概述 


文件 下 载 是 上 传 与 下 载 模式 的 另 一 核心 内 容 。 首 先 ， 无 论 用 户 登 录 与 否 ， 都 可 以 对 已 公开 的 文件 
进行 下 载 : 当 用 户 登录 后 ， 不 但 可 以 上 传 新 的 文件 ， 同 时 也 可 以 对 已 经 上 传 的 文件 执行 下 载 操 作 和 文 
件 删除 操作 ;如 果 当 前 用 户 没有 登录 ， 作 为 游客 身份 ， 也 可 以 通过 获得 与 文件 匹配 的 下 载 码 ， 对 不 公 
开 文件 执行 下 载 操作 。 文 件 下 载 页 的 运行 结果 如 图 3.15 所 示 。 


名 
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图 3.15 文件 下 载 页 的 运行 结果 
3.7.2 通过 下 载 码 下 载 文件 


用 户 登录 后 ， 可 以 通过 “下 载 文件 ” 超 链接 或 输入 下 载 码 对 文件 进行 下 载 。 这 里 以 下 载 码 下 载 为 
例 ， 讲 解 文件 下 载 的 实现 过 程 。 

下 载 码 下 载 的 操作 在 index.php 文件 中 完成 。 首 先 ， 创 建 下 载 码 文本 框 ， 提 交 下 载 码 ， 其 关键 代码 
如 下 : 


<div id="found">&nbsp; 

下 载 码 : 

<input id="chkdown" type="text" />&nbsp; 

<button id="downbtn" onclick="chkdown()"></button> 


在 index.php 文件 中 ， 当 用 户 单 击 “ 下 载 ” 按 钮 时 ， 将 触发 onclick 事件 ， 调 用 chkdownO) 函 数 判断 
输入 值 是 否 为 室 ， 如 果 不 为 空 ， 将 重新 加 载 本 页 ， 并 将 下 载 码 作为 参数 传递 。chkdown(O) 函 数 存储 于 
js\indexjs 脚本 文件 中 ， 其 关键 代码 如 下 : 


function chkdown(X{ 

if($('chkdown').value == "){ 

alert(' 请 输入 下 载 码 !"); 

return false; 

} 

location = ?act=downcode&chkdown='+$('chkdown').value; 
} 


然后 ， 在 index.php 页 中 ， 当 超 链接 传递 的 值 等 于 downcode 时 ， 加 载 downcode.php 页 。 
<!-- 根 据 不 同 的 参数 值 ， 载 入 不 同 的 页 面 > 


<div id="content"> 
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<?php 

$act = $_GET[act]; 

Switch($actf 

case 'up': 

include "upfile.php'; / 载 入 上 传 页 

break; 

case 'upfile': 

include "upfile_chk.php'; 1// 载 入 上 传 处 理 页 

break; 

case 'down': 
include 'downfile.php'; // 载 入 下 载 页 
break; 

case 'downcode': 

include 'downcode.php'; // 载 入 下 载 码 下 载 页 

break; 

default: 

include 'pub.php'; // 载 入 共享 文件 页 
break; 

} 

?> 

</div> 


接着 , 创建 downcode.php 页 ， 根据 获取 的 下 载 码 生成 SQL 查询 语句 ,循环 输出 下 载 文件 信息 ， 并 
且 创 建文 件 下 载 超 链接 ， 链 接 到 download.php 文件 ， 完 成 文件 下 载 操作 。 关 键 代码 如 下 : 


<?php 
$chkdown = trim($_GET[chkdown]); 
$downsql = "select * from tb_upfile where chkdownnum = ".$chkdown.""; 
$downarr = $conne->getRowsArray($downsql); 
?> 
<!-- 下 载 文件 列表 -> 
<div id="filelist"> 
<?php 
$infoline = 0; 
if(lempty($downarr))f // 如 果 结果 集 数组 不 为 空 
?> 
<ul> 
<li style=" width: 30px; background-color #F8E7AS;">&nbsp;</li> 
<li style=" width: 200px; background-color#F8E7A5; "> 文件 名 </li> 
<li style=" width: 50px; background-color:#F8E7A5; "> 文件 类 型 </li> 
<li style=" width: 100px; background-color:#F8E7A5;"> 下 载 码 </li> 
<li style=" width: 150px; background-color:#F8E7A5;"> 上 传 时 间 </li> 
</ul> 
<?php 
foreach($downarr as $key => $valueX{ // 循 环 输出 数组 
?> 
<ul> 
<li style=" width: 30px;background-color: <?php echo (S$infoline %2 ==0?'#FFFFFF":#f0f0f0"); ?>;"> 
<input id="chk[<?php echo $key; ?>]" type="checkbox" value="<?php echo $value['id]; ?>" 
style=" width:30px; height:20px; border Opx;" /></li> 


多 
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<li style=" width: 200px; background-color:<?php echo (S$infoline %2 ==0?'#FFFFFF"':#f0f0f0"); ?>;"> 
&nbsp; <a href="download.php?id=<?php echo $value[id]; ?>"><?php echo $value[filename]; ?></a></li> 

<li style=" width: 50px; background-color: <?php echo (S$infoline %2 ==0?'#FFFFFF"':#f0f0f0"); ?>;"> 
&nbsp; <?php echo $value[filetype']; ?></li> 
<li style=" width: 100px; background-color <?php echo ($infoline %2 ==0?#FFFFFF":#f0f0f0"); ?>;"> 
&nbsp; <?php echo $value[chkdownnum]; ?></li> 

<li style=" width: 150px; background-color: <?php echo (S$infoline %2 ==0?#FFFFFF":#f0f0f0"); ?>;"> 
&nbsp; <?php echo $value[uptime]; ?></li> 


</ul> 

<?php 

S$infoline = ($infoline+1)%2; 

} 

}else{ 

echo ' 没 有 可 下 载 资源 '; 

} 

?> 

</div> 

最 后 ， 创 建 download.php 文件 ， 根 据 文件 的 完整 路 径 和 header() 函 数 完 成 下 载 。 关 键 代 码 如 下 : 
<?php 

include_once 'conn/conn.php'; 

$id =$_GET[id]; // 获 取 用 户 的 id 

$downsql = "select * from tb_upfile where id = $id"; // 通 过 where 子 句 获得 相应 的 文件 
$downarr = $conne->getRowsRst($downsql); 

$path=$downarr[filepath ]; // 获 取 路 径 名 称 赋 给 $path 


if(lempty($path) and !is_null($path)}{ 
$filename=basename($path); 
S$file=fopen( $path,"r"); 
header("Content-type:application/octet-stream"); 
header("Accept-ranges:bytes"); 
header("Accept-length:".filesize($path)); 
header("Content-Disposition:attachment;filename=". $filename); 
echo fread($file,filesize($path)); 
fclose($file); 
exit; 
二 


?> 


3.7.3 同步 删除 文件 以 及 释放 用 户 空间 


单 击 “ 下 载 文件 ” 超 链接 后 会 在 下 载 文件 的 下 方 看 到 删除 选择 的 文件 ， 在 删除 数据 库 文件 的 同时 ， 


也 会 删除 该 登录 用 户 相应 文件 夹 下 的 同名 文件 ， 并 且 同 步 更 新 用 户 使 用 空间 的 大 小 ， 达 到 删除 文件 释 
放空 间 的 效果 ， 删 除 文件 后 ，1 秒 钟 之 后 返回 下 载 首页 。 删 除 文件 释放 空间 的 操作 在 downfile.php 文件 
中 执行 ， 其 关键 代码 如 下 : 


<?php 
S$del = $_GET[del]: 
$num =10; /每 页 显示 记录 数 


@ 
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$curpage = $_GET[curpage]; // 当 前 页 

$sql = "select * from tb_upfile where upauthor = ".$_SESSION[name].”"; 

S$totnum = $conne->getRowsNum($sql); /| 记录 数 
if($del == 'del)f 

$rd=$_GET[rd]; 

$delsql = "delete from tb_upfile where id = -1"; 

$deltotal = "select sum(filesize) as total from tb_upfile where id = -1 "; // 取 得 选择 文件 的 总 大 小 
$content="select * from tb_member where name =".$_SESSION[name].”"; 

$contarr = $conne->getRowsRst($content); 

if(lempty($rd)\{ 

$delarr = explode(',',$rd); 

foreach($delarr as $value){ 

// 以 下 为 新 加 内 容 ， 在 删除 文件 的 同时 也 可 以 删除 硬盘 文件 夹 上 的 文件 

$sql= "select * from tb_upfile where id = ".$value." "; /选取 删除 文件 的 id 值 
$delsql .= "orid = ".$value." "; 

$deltotal .= " or id = ".$value.”"; 


$sqlarr = $conne->getRowsRst($sql); // 调 用 SQL 函数 
$delpath=$sqlarr[filepath1]; // 获 得 删除 文件 路 径 
unlink($delpath); /删除 指定 路 径 的 文件 


$deltotalnum=$conne->getRowsRst($deltotal); 

$newdata = $contarrf' content]-$deltotalnum['otal]; 

// $newdata 为 删除 文件 后 个 人 空间 的 使 用 大 小 

$newsql="update tb_member set content = $newdata where name ='".$_SESSION[name].”"; 
$conne->uidRst($newsql); /把 删除 文件 后 剩余 空间 的 大 小 更 新 到 数据 库 
$delnum = $conne->uidRst($delsql); 

echo "<meta http-equiv=\"refresh\" content=\"1;url=index.php?act=down\">";”//1 秒 钟 之 后 跳 转 到 主页 
echo "<p></p>"; 

echo "删除 成 功 ! "; 


3.8.1 后 台 管 理 概述 


在 后 台 管 理 系统 中 只 有 一 个 功能 ， 就 是 管理 注册 用 户 的 权 


限 。 其 运行 结果 如 图 3.16 所 示 。 用 户 权限 管理 
目前 共有 2 条 记录 共 分 ! 页 显示 ”当前 显示 第 1 页 
3.8.2 用户 权限 管理 页 面 设 计 用 户 各 。 合用 机 限 合用 窑 量 。。 修 疏 


mrsoft ”高 级 用 户 0. OOMB 修改 
leonsk 高 级 用 户 0.00WB 全 发 


进入 到 后 台 管 理 系 统 中 ， 以 分 页 形式 输出 已 经 注册 的 用 户 第 -页 | 上 -页 | 下 -页 | 最 后 -页 


信息 ， 包 括 用 户 名 、 使 用 权限 和 使 用 容量 ， 以 及 “修改 ” 超 链 


接 。adminphp 的 关键 代码 如 下 : 0 


<?php 
include_once "conm/conn.php'; 


@. 
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Slist_num=10; 

$sql = "select * from tb_member"; /查询 语句 
$sqlarr = $conne->getRowsRst($sql); // 返 回 查询 结果 集 
$totnum = $conne->getRowsNum($sql); // 获 取 查 询 记录 数 
$conne->close_rst(); 

if($totnum>0){ 

?> 

<p> 

<table> 

<tr class="tr1"> 

<td> 用 户 名 </td> 

<td> 使 用 权限 </td> 

<td> 使 用 容量 </td> 

<td> 修 改 </td> 

</tr> 

<?php 

$sql = $sql." order by id desc limit ".($list_num *($page-1)).",".$list_num; // 查 询 语句 
S$filearray = $conne->getRowsArray($sql); 

foreach($filearray as $valueX{ 

echo "<tr>\n"; 

echo "<td>".$value[name']."</td>\n"; 

switch($value[right])f // 查 看 用 户 权限 
case '0': 

echo "<td> 普 通用 户 </td>\n"; 

break; 

case '1': 

echo "<td> 会 员 用 户 </td>\n"; 

break; 

Case '2 7 

echo "<td> 高 级 用 户 </td>\n"; 

break; 

} 

echo "<td>".sprintf("%-01.2f",($value['contentY/1000000))."MB</td>\n"; 

echo "<td><a href=changesub.php?id=".$value[id]."> 修 改 </a></td>\n"; /获取 id 更 改 用 户 权限 
echo "</tr>\n"; 


echo "</table>"; 


3.8.3 ”更 改 用 户 权限 表单 设计 


更 改 用 户 权限 从 管理 页 中 设置 的 “修改 ” 超 链接 开始 ， 在 changesub.php 文件 中 ， 根 据 超 链接 传递 
的 ID 值 ， 查 询 出 指定 用 户 的 详细 信息 ， 并 且 将 该 信息 在 表单 中 输出 ， 进 而 通过 表单 将 数据 提交 到 指定 
页 完成 对 用 户 权限 的 修改 操作 。changesub.php 文件 的 关键 代码 如 下 : 


<table border=1> 

<form action="change.php" method="post"> 
<?php 

include_once 'conn/conn.php'; 


®@ 


PHP 典型 模块 开发 全 程 实录 


$sql="select * from tb_member where id = $id"; /查询 语句 
$changearr = $conne->getRowsRst($sql); // 返 回 查询 结果 
?> 

<tr> 

<td> 用 户 名 </td> 

<td><?php echo $changearr[name];?></td> 

</tr> 

<tr> 

<td> 当 前 权限 </td> 

<td> 

<?php 

switch($changearr[right])f // 获 取 用 户 权 限 
case '0": 

echo "普通 用 户 "; 

break; 

case 人 

echo "会 员 用 户 "; 

break; 

Case '2": 

echo "高 级 用 户 "; 

break; 

} 

?> 

</td> 

</tr> 

<tr> 

<td> 修 改 权限 </td> 

<td> 

<select name="right"> // 选 择 用 户 权 限 
<option value="0" selected="selected"> 普 通用 户 </option> 

<option value="1"> 会 员 用 户 </option> 

<option value="2"> 高 级 用 户 </option> 

</select> 

</td> 

</tr> 

<tr><td colspan=2> 

<center> 

<input type="submit" value=" 提 交 " /> 

</center> 

</td> 

</tr> 

<input type="hidden" name="id" value="<?php echo $changearrfid]?>" />// 添 加 一 个 隐藏 域 传递 id 参数 
</form> 

</table> 


3.8.4 执行 权限 更 改 操作 


创建 在 changesub.php 中 指定 的 权限 更 改 处 理 文件 change.php， 用 于 获取 表单 中 提交 的 数据 ， 完 成 
用 户 权 限 的 更 改 操作 。change.php 文件 的 关键 代码 如 下 : 


@. 
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<?php 
include_once "conn/conn.php'; 
$right=$_POST[right]; // 获 得 权限 更 改 表单 的 传递 值 


$id=$_POST[id]; 

$upsql = "update tb_member set tb_member.right =$right where id = $id"; /查询 语句 
$num=$conne->uidRst($upsql); /返回 结 果 
$conne->close_rst(); 

echo "<p>"; 

if($num==1){ 

echo "权限 更 改 成 功 !"; 

echo "<p>"; 

}else{ 

echo "权限 更 改 失 败 !"; 

echo "<p>"; 

六 

?> 

<! 一 以 下 为 html 内 容 ， 修 改 成 功 会 返回 修改 主页 -> 

<meta http-equiv="refresh" content="2;url=admin.php" /> 

2seconds laterto return the manger page! 


3.9 技术 提炼 


3.9.1 ”php.ini 配置 文件 


如 果 要 在 PHP 中 实现 小 文件 的 上 传 (2MB 以 下 ) ， 那 么 无 须 对 php.ini 配置 文件 进行 修改 ， 使 用 
默认 参数 即 可 。 但 如 果 想 实现 完美 的 上 传 功能 ， 则 一 定 要 对 配置 文件 有 所 了 解 。 在 php.ini 文件 中 ， 与 
文件 上 传 有 关 的 设置 如 图 3.17 所 示 。 


244 ; Maximum execution time of each script, in seconds 


rer 

246 2 Maximu amounc of cime each script may spend parsing request data 
Te 

248 ; Haximum amount Of memory a Script may consume (16MB) 


za 
zsd 


a11|; Naximum size of POS] data that PHP will accept. 
max, 16N 


re 
| 


3.17 php.ini 文件 中 的 相关 参数 
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从 图 3.17 中 可 以 看 到 ， 有 3 部 分 内 容 需 要 进行 配置 ， 分 别 介绍 如 下 。 
1. Resource Limits 
Resource Limits， 直 译 过 来 就 是 资源 限制 ， 包 含 了 3 个 参数 。 该 区 块 不 仅仅 是 针对 上 传 与 下 载 的 ， 
而 是 对 全 部 的 文件 进行 设置 。 各 个 参数 含义 及 参数 值 说 明 如 表 3.1 所 示 。 
表 3.1 Resource Limits 块 的 参数 说 明 


描述 
每 个 脚本 页 面 完成 执行 操作 的 最 大 时 间 ， 单 位 是 秒 。 如 果 设 为 -1， 说 明 没有 限制 
每 个 脚本 页 面 处 理 请 求 数据 的 最 大 时 间 ， 单 位 是 秒 ， 也 可 以 设 为 -1 
一 个 脚本 页 所 能 够 消耗 的 最 大 内 存 


参数 


max_execution_time 


max input time 


memory_limit 


2. post max size 
post_max_size 参数 指 PHP 通过 表单 POST 所 能 接收 的 最 大 值 ， 包 括 表 单 中 的 所 有 项 。 
3. File Uploads 


File Uploads 块 是 专 为 文件 上 传 设置 的 。 该 区 块 中 同样 包含 了 3 个 参数 。 参 数 含义 及 参数 值 说 明 如 
表 3.2 所 示 。 


表 3.2 File Uploads 块 的 参数 说 明 


file uploads 是 否 允许 HTTP 上 传 ， 默 认为 On， 即 为 开启 ， 无 须 修改 


upload tmp_dir 文件 上 传 时 的 临时 存储 目录 。 如 果 没 指定 就 会 用 系统 默认 的 临时 文件 夹 
upload_max_filesize 允许 上 传 文件 的 最 大 值 


3.9.2 上传 表单 设置 


配置 文件 修改 完毕 后 , 重启 Apache 服务器， 接 下 来 开始 设计 <form> 表 单 。<form> 表 单 可 以 实现 人 
机 交互 ， 是 动态 网 页 的 一 个 最 基本 功能 。 最 初 的 <form> 表 单 不 支持 文件 上 传 ， 表单 中 input 元 素 的 type 
属性 只 有 7 种 值 : 文本 框 (text) 、 密 码 框 〈text) 、 单 选 按钮 (radio) 、 复 选 框 (checkbox) 、 重 置 
按钮 (reset) 、 确 定 按钮 (submit) 和 图 像 (image) 。 后 来 ，rfe1867 为 HTTP 协议 添加 了 这 个 功能 ， 
目前 儿 乎 所 有 的 浏览 器 和 服务 器 都 支持 此 协议 。 

rfc1867 协议 为 type 属性 增加 了 file 值 ， 同 时 ，<form> 标 签 中 的 enctype 属性 值 必须 设 为 
multipart/form-data，method 属性 则 必须 是 POST，3 个 条 件 缺 一 不 可 。 当 表单 属性 设置 正确 后 ， 才 可 以 
在 表单 处 理 页 中 获取 到 上 传 文件 的 值 。 下 面 是 简单 、 可 用 的 上 传 表单 代码 。 


<form method="post" action="2.php" enctype="multipart/form-data"> 
<input name="filename" type="file" /> 
<input type="submit" value="enter"/> 

</form> 


此 外 , 还 建议 在 上 传 表单 中 增加 一 个 hidden 隐藏 域 , 隐藏 域 的 name 值 为 MAX_FILE_SIZE, value 


@ 
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值 是 允许 上 传 的 最 大 字 节 数 。 例 如 : 
<input type="hidden" name="MAX_FILE_SIZE" value="1024000" /> 


这 样 ， 当 上 传 文件 大 于 1MB 时 ，$_FILES 中 的 error 值 将 返回 2。 关 于 $_FILES 变量 的 使 用 在 3.9.3 
节 中 给 出 。 该 隐藏 域 的 作用 不 是 真正 要 去 限制 上 传 文件 的 大 小 ， 因 为 要 绕 过 该 项 检查 非常 简单 ， 它 的 
真正 作用 是 为 了 避免 用 户 误 传 了 大 文件 而 陷入 “无 尽 的 ”等 待 之 中 。 


a 
3.9.3 ”预定 义 变量 $_FILES 


如 果 表 单 设计 得 没有 问题 ， 那 么 在 处 理 页 中 ， 使 用 $_FILES 预定 义 变量 就 可 以 得 到 上 传 文件 的 相 
关 信 息 ; 反之 ， 如 果 得 不 到 ， 那 么 就 有 可 能 是 配置 文件 或 表单 设置 出 问题 了 。 下 面 先 介绍 预定 义 变量 
$_FILES 。 

$_FILES 变量 存储 的 是 上 传 文件 的 相关 信息 ， 这 些 信息 对 于 上 传 功能 有 很 大 作用 。 该 变量 是 一 个 
- 维 数组 ， 保 存 的 信息 如 表 3.3 所 示 。 


表 3.3 $_FILES 变量 值 


参数 描述 
$_ FILES[filename]['name' 存储 了 上 传 文件 的 文件 名 。 例 如 exam.txt、myDream.jpg 等 
$_FILES[filenamejl['size' 存储 了 文件 大 小 。 单 位 为 字 节 
$_FILES[filenamel[tmp_name' 文件 上 传 时 ， 首 先 在 临时 目录 中 被 保存 成 一 个 临时 文件 。 该 变量 为 文件 名 
$ FILES[filenamel['type' 上 传 文件 的 类 型 。 注 意 是 类 型 ， 不 是 后 级 
$_ FILES[filenamel['error 存储 了 上 传 文件 的 错误 代码 


error 字段 将 返回 上 传 文件 的 错误 码 ， 一 共有 7 种 状态 。 错 误 代码 及 说 明 如 表 3.4 所 示 。 

表 3.4 $_FILES[filename][eror] 中 的 错误 代码 及 说 明 
错误 代码 错误 说 明 
| UPLOAD_ ERR OK。 上 传 成 功 ， 无 错误 
| UPLOAD_ERR_INI_SIZE。 文件 大 小 超出 php.ini 中 upload_max_filesize 的 参数 值 
| UPLOAD_ERR_FORM_SIZE。 文 件 大 小 超出 表单 中 MAX_FILE_SIZE 隐藏 域 的 值 
| 
| 
| 


UPLOAD ERR_PARTIAL。 文件 没有 被 全 部 上 传 
UPLOAD ERR _NO FILE。 没 有 文件 被 上 传 
UPLOAD _ERR_NO_TMP_DIR。 找 不 到 临时 文件 目录 


3 UPLOAD ERR CANT WRITE。 文件 写 入 失败 


下 面 使 用 3.6.2 节 中 的 表单 上 传 一 个 文件 ， 使 用 $_FILES 变量 输出 文件 信息 。 代 码 如 下 : 


<?php 
S$tmparr = $_FILES['filename'; /获取 文件 域 的 值 
if($tmparr[rname]!= "X{ /| 如果 上 传 文件 名 不 为 空 


@ 
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foreach($tmparr as $name => $value\{ 

echo $name.' => '.$value.'<br>"; // 使 用 循环 输出 数组 索引 和 值 
}else{ 

echo "<script>alert(' 请 添加 上 传 文件 ');history.go(-1);</script>"; 

} 

?> 

运行 结果 如 图 3.18 所 示 。 


| 


pplicetion/octet-stre 
tmp_name => 了 te 下 tp 
error => 0 


size =》1337914 


图 3.18 ”获取 上 传 文件 的 信息 


3.9.4 ”上 传 文件 函数 

move_uploaded_file() 函 数 用 来 上 传 文件 到 指定 的 位 置 。 如 果 成 功 返 回 true， 否 则 返回 false。 该 函 
数 的 语法 如 下 : 

bool move_uploaded file ( string filename, string destination ) 


filename: 上 传 文件 的 临时 文件 名 ， 即 $_FILES['tmp_name']。 

destination: 上 传 后 保存 的 新 的 路 径 和 名 称 。 

通过 3.6.3 节 和 3.7.3 节 的 阅读 ， 细 心 的 读者 可 能 已 经 发 现 ， 真 正 的 上 传 文件 已 经 被 保存 到 临时 目 
录 中 ，move_uploaded_file() 函 数 的 作用 只 是 将 文件 换个 地 方 保 存 ， 这 是 出 于 安全 的 考虑 。 

此 外 ， 还 可 以 通过 is_uploaded_file0) 函 数 判断 文件 是 否 为 POST 上 传 。 该 函数 的 语法 如 下 : 


bool is_uploaded file ( string filename ) 


3.9.5 多 文件 上 传 


多 文件 上 传 和 单 文件 上 传 没什么 区 别 ， 只 要 将 表单 中 的 文件 域 用 数组 命名 即 可 。 代 码 如 下 : 
<form action="2.php" method="post" enctype="multipart/form-data"> 

<input name="userfile[" type="file" /><br /> 

<input name="userfile[]" type="file" /><br /> 


<input type="submit" value=" 上 传 " /> 
</form> 


@ 
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上 传 的 多 文件 可 以 通过 for 循环 语句 进行 获取 。 方 法 如 下 : 


$filename= $_FILES[userfilel[name']; 


for ($i = 0; $i < count($filename); $i++\{ // 通 过 循环 将 多 个 文件 全 部 上 传 
move_uploaded file($tmpname[$j,$file_path.$filename[$ij); 
// 添 加 数据 库 


$insertsql = 'insert into tb_upfile (filename,filepath,filetype,upauthor,chkdownnum,ispub) 
values(".trim($filename[$1]).",".S$file_path. $filename[$i].",". $filetype[$i].", 
".$_SESSION[name'].",".tim($chkdownnum).",".$tmppub[$i].")'; 
$conne->uidRst($insertsql); 

echo '<div style=" padding-top: 30px;"> 文 件 上 传 成 功 

<font color="#FF0000">'.$chkdownnum.'</font>&nbsp; 对 文件 进行 下 载 。</div>'; 


7 
TT 
件 时 ， 多 个 文件 会 获取 相同 的 下 载 码 。 在 上 传 多 个 文件 时 ,判断 多 个 文件 上 传 是 否 有 重 名 比较 麻烦 ， 
一 旦 上 传 失 败 ， 会 导致 所 有 文件 都 上 传 失 败 ， 必 须 重新 操作 上 传 步 又。 这 时 可 以 在 上 传 页 面 增加 
JavaScript 脚本 ， 在 提交 之 前 判断 是 否 存在 同名 文件 ， 如 果 不 加 入 这 个 判断 过 程 ， 在 其 后 出 除 多 个 
同名 文件 时 就 会 导致 出 错 ， 有 兴趣 的 读者 可 以 自行 研究 一 下 . 


3.9.6 文件 下 载 


在 一 个 完整 的 网 站 中 ， 拥 有 文件 下 载 功能 能 够 给 网 站 带 来 更 多 的 访问 者 ， 增 加 网 站 的 访问 量 。 实 
现下 载 最 常用 的 方式 有 通过 链接 下 载 和 通过 headerO 函 数 下 载 。 

(1) 通过 链接 下 载 

通过 链接 下 载 ， 就 是 将 文件 的 相对 路 径 或 绝对 路 径直 接 作为 超 链 接 即 可 。 如 果 是 压缩 文件 ， 那 么 直 
接 单 击 超 链接 就 可 以 下 载 ; 对 于 文本 文件 ， 单 击 鼠标 右键 ， 在 弹出 的 快捷 菜单 中 选择 “目标 另存 为 ” 命 
令 ， 就 可 以 将 文本 文件 下 载 下 来 。 所 有 格式 的 文件 都 可 以 通过 这 种 方法 下 载 ， 运 行 结果 如 图 3.19 所 示 。 


图 3.19 通过 链接 下 载 


(2) 通过 header() 函 数 下 载 
header() 函 数 属于 HTTP 函数 ， 它 的 作用 是 发 送 一 个 原始 的 HTTP 标 头 。header(O) 函 数 的 语法 如 下 : 
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void header ( string string [, bool replace [, int http_response_code]] ) 

string: 发 送 的 标 头 。 

回 ”replace: 如 果 一 次 发 送 多 个 标 头 ， 指 明 相 似 的 标 头 是 替换 还 是 添加 。 如 果 是 包 lse， 则 强制 发 
送 多 个 同类 型 标 头 。 默 认 是 tue， 即 替换 。 

http_response_code: 强制 HTTP 响应 为 指定 值 。 

HTTP 标 头 有 很 多 ， 这 里 只 介绍 下 载 ， 其 他 请 参考 相关 资料 。 下 载 HTTP 标 头 的 代码 如 下 : 


header('Content-Disposition: attachment; filename="filename"™); 


上 面 的 代码 中 唯一 需要 改动 的 就 是 filename。 将 filename 替换 为 要 下 载 的 文件 就 可 以 完成 下 载 。 
3.10 本 章 总 结 


本 章 通 过 详细 讲解 一 个 完整 的 上 传 与 下 载 模块 。 通 过 该 模块 可 以 将 要 上 传 的 工具 、 文 档 、 音 频 或 
视频 等 多 种 类 型 的 文件 上 传 到 个 人 空间 中 。 文 件 下 载 是 上 传 与 下 载 的 另 一 核心 内 容 。 首 先 ， 无 论 用 户 
登录 与 否 ， 都 可 以 对 已 公开 的 文件 进行 下 载 ， 当 用 户 登 录 后 ， 不 但 可 以 上 传 新 的 文件 ， 同 时 也 可 以 对 
已 经 上 传 的 文件 执行 下 载 操 作 和 文件 删除 操作 。 


全 立 
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FTP 管理 工具 作为 一 种 服务 器 管理 系统 ， 其 为 管理 者 通过 互联 网 
执行 远程 操作 或 管理 服务 器 带 来 了 极 大 方便 。FTP 协议 是 Internet 上 
的 文件 传输 通用 协议 ， 它 位 于 TCP/IP 协议 的 应 用 层 。 利 用 该 协议 ， 用 
户 可 以 将 一 个 完整 的 文件 从 一 各 计 算 机 复制 到 另 一 各 计算 机 上 。 但 是 
在 使 用 FTP 传输 文件 之 前 ， 需 要 登录 FTP 服务 器 ， 用 户 可 以 通过 已 设 
置 的 FTP 用 户 名 和 密码 登录 服务 器 。 本 章 向 读者 展现 一 个 多 功能 FTP 
管理 系统 ， 主 要 实现 本 地 系统 与 远程 FTP 间 文 件 的 上 传 下 载 和 管理 文 
件 等 功能 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

| 遍历 FTP 服务 路 目录 

MW 获取 本 地 文件 的 图 标 

WI 实现 FTP 文件 上 传 下 载 

本 地 文件 与 FTP 文件 的 维护 操作 
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4.1 FTP 管理 系 统 概 述 


4.1.1 模块 概述 


FTP 管理 系统 是 一 个 可 以 执行 常用 操作 的 上 传 下 载 工 具 ， 其 作用 是 方便 用 户 快 捷 地 访问 FTP 服务 
器 , 应 用 FTP 传输 文件 的 独立 性 。 用 户 与 用 户 在 分 别 执行 各 自 的 上 传 、 下 载 和 维护 等 操作 时 互 不 影响 。 
另外 ，FTP 管理 系统 具有 良好 的 远程 管理 性 ， 用 户 可 以 通过 互联 网 登录 FTP 服务 器 ， 执 行 远程 管理 
操作 。 


4.1.2 功能 结构 
FTP 管理 系统 包括 根 目录 操作 和 子 目录 操作 两 大 基本 结构 。 其 中 根 目录 操作 包括 查看 文件 、 查 看 


目录 、 删 除 目录 、 创 建 目录 和 上 传 文件 ; 子 目录 操作 包括 查看 文件 、 查 看 目录 、 更 改 文件 、 移 动 文件 、 
删除 文件 、 创 建 目 录 和 上 传 文件 。FTP 管理 系统 的 功能 结构 如 图 4.1 所 示 。 


子 目录 操作 


根 目录 操作 


图 4.1 FTP 管理 系统 的 功能 结构 


4.1.3 程序 预览 
为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 几 个 页 面 运行 效果 图 ， 如 果 想 查 


看 完整 的 效果 图 ， 请 参见 光盘 源 程序 。 
FTP 管理 系统 的 登录 页 面 和 主页 面 的 运行 效果 如 图 4.2 和 图 4.3 所 示 。 


@ 
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图 4.2 登录 页 面 的 运行 效果 


StD 服 务 器 的 换 作 系统 是 WWIX 。。 当前 时 间 : 2011 年 05 月 09 日 六 创建 目录 上 传 文件 


., 昌 日 FTP 管 理 系统 V1.0 


图 4.3 FTP 主页 面 的 运行 效果 
子 目录 下 的 文件 详细 信息 运行 结果 如 图 4.4 所 示 。 


当前 目录 : /AppSerw/Len0402/ 
创建 目录 上传 文件 


文件 名 称 文件 大 小 景 后 修改 时 间 更 改名 称 ”删除 文件 “移动 文件 


acc_manager_2. php 3.5K 2011-05-03 10:37:29 更 改 删除 移动 


图 4.4 子 目录 下 的 文件 详细 信息 运行 结果 
单 击 文件 名 称 的 超 链 接 ， 下 载 FTP 文件 的 运行 结果 如 图 4.5 所 示 。 单 击 “ 移 动 ” 超 链接 ， 移 动 文 


件 到 指定 文件 目录 的 操作 如 图 4.6 所 示 。 


时 PHP 典型 模块 开发 全 程 实录 


将 /hppServ/Len0402/acc_manager_2. php 移 动 到 : 


d: \A ALenn0402 
Se AppServy 


# 青 填写 有 效 路 径 ( 如 C:\LeonkAppserv\) + 请 卦 写 正确 的 路 径 。" 如 /appservyf 


下 载 | | 重 置 


[移动 文 伞 ] 


图 4.5 下 载 FTP 文 件 的 运行 结果 图 4.6 移动 文件 到 指定 文件 目录 的 操作 
4.2 FTP 环境 搭建 


4.2.1 FTP 服务 器 端 概述 


FTP 服务 器 端 是 一 种 可 以 将 计算 机 或 服务 器 设置 成 FTP 服务 器 的 软件 工具 。 通 过 FTP 软件 ， 用 户 
或 其 他 使 用 者 就 能 够 使 用 FTP 协议 ， 通 过 互联 网 或 者 同一 网 络 上 的 任何 一 台 计算 机 与 FTP 服务 器 连 
接 ， 执 行文 件 或 目录 的 复制 、 移 动 、 创 建 和 删除 等 操作 。 这 里 提 到 的 FTP 协议 是 专门 被 用 来 规定 计算 
机 之 间 进 行文 件 传输 的 标准 和 规则 ， 正 是 因为 有 了 像 FTP 这 样 的 专门 协议 ， 才 使 得 人 们 能 够 通过 不 同 
类 型 的 计算 机 ， 使 用 不 同类 型 的 操作 系统 ， 实 现 不 同类 型 的 文件 进行 相互 传递 。 

目前 FTP 软件 的 种 类 繁多 ， 其 功能 各 有 千秋 。 其 中 Serv-U 是 应 用 最 为 广泛 的 FTP 软件 之 一 ， 其 
凭借 优秀 的 功能 博得 人 们 的 青睐 。Serv-U 软件 可 以 提供 以 下 功能 : 
符合 Windows 标准 的 用 户 界面 友好 亲切 ， 易 于 掌握 。 
通过 限制 同一 时 间 最 大 的 用 户 访问 人 数 确保 PC 的 正常 运转 。 
安全 性 能 出 众 。 在 目录 和 文件 层次 中 都 可 以 设置 安全 防范 措施 。 
能 够 为 不 同 用 户 提供 不 同 设置 ， 支 持 分 组 管理 数量 众多 的 用 户 。 
可 以 基于 IP 对 用 户 授予 或 拒绝 访问 权限 。 
支持 文件 上 传 和 下 载 过 程 中 的 断 点 续 传 。 
支持 拥有 多 个 IP 地 址 的 多 宿主 站 点 。 
能 够 设置 上 传 和 下 载 的 比率 、 硬 盘 空 间 配额 、 网 络 使 用 带宽 等 ， 从 而 能 够 保证 用 户 有 限 的 资 
源 不 被 大 量 的 FTP 访问 用 户 所 消耗 。 
可 作为 系统 服务 后 台 运 行 。 
[自行 设置 在 用 户 登 录 或 退出 时 的 显示 信息 ， 支 持 具有 UNIX 风格 的 外 部 链接 。 


4.2.2 Serv-U 的 安装 和 配置 


为 了 保证 FTP 管理 系统 能 够 正常 运行 并 发 挥 其 应 有 的 功能 效果 ， 在 使 用 FTP 管理 系统 前 应 在 远 
程 服务 器 上 搭建 FTP 服务 环境 。 本 系统 选择 Serv-U 来 搭建 FTP 服务 环境 。FTP 服务 环境 的 架设 过 程 
如 下 : 

(1) 用 户 可 以 通过 互联 网 下 载 Serv-U FTP Server 安装 包 。 将 下 载 好 的 安装 解压 缩 到 指定 文件 目 


办 办 办 办 办 欠 轨 


| 避 


加 回 
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录 中 ， 并 双击 ServUSetup.exe 打开 安装 文件 。 

(2) 选择 语言 为 中 文 (简体) 形式 ， 并 单 击 “ 确 定 ” 按 钮 进入 下 一 步 安装 向 导 设置 。 安 装 向 导 运 
行 界面 如 图 4.7 所 示 。 

(3) 单 击 “ 下 一 步 ”按钮 ， 进 入 许可 协议 查看 界面 。 选 择 “ 我 接受 协议 ”后 ， 继 续 单 击 “ 下 一 步 ” 
按钮 ， 进 入 选择 目标 位 置 界 面 ， 如 图 4.8 所 示 。 


Ce Ee | 
图 Serv-U 次 迎 使 用 Serv-U 安装 向 导 | Res © 
即 相安 装 Serv 14.0 到 电脑 上 。 
ee .| 有 Sie sur 安 半 lbAT 文 件 中 。 | 


| 车 要 继续 ， 单 击 “ 下 一 步 ”。 如 果 你 要 选择 不 同 的 文件 去， 请 单 击 “ 浏 览 ”* 


站 Eee 3 


单 击 “ 下 一 步 ”继续 ， 或 单 击 “ 取 消 " 退出 安装 。 


至 少 需要 36. 1 好 的 空 用 磁盘 空间 。 


CE-SY FBV >) (WN 


[ED | 


图 4.7 安装 向 导 运 行 界面 图 4.8 选择 目标 位 置 界面 
(4) 设置 路 径 完 毕 后 ， 继 续 单 击 “ 下 一 步 ”按钮 ， 进 入 准备 安装 界面 ， 如 图 4.9 所 示 。 单 击 “ 安 
装 ” 按 钮 开始 安装 软件 。 
(5) 当 软 件 的 所 有 部 分 安装 完毕 后 ， 弹 出 如 图 4.10 所 示 的 界面 。 


S 到 有 安 半 内 导 - Serv-U RE | 
闪失 sur @ | 多 Serv-U 完成 serv-uU 安 半 
点 击 “ 安 装 " 继续 安装 ， 如 果 俐 起 要 埋 看 或 者 更 次 讼 轩 请 点 击 “上 一 步 "。 A 
BH Survserry 点 击 “ 守 所" 退守。 
开 妈 芝 间 文 从: 避 启动 Serv 管理 控制 台 


目 查 看 srt 用 户 手册 
了 入: ev 作为 系统 务 安装 


ZEg]CESEO [一 
图 4.9 准备 安装 界面 图 4.10 ”安装 完成 


这 样 FTP 服务 器 端 软件 Serv-U 即 安装 完毕 。 用 户 单 击 “ 完 成 ”按钮 来 运行 Serv-U 软件 ， 系 统 会 
提示 用 户 设置 相应 的 必要 参数 。 下 面 讲解 如 何 配置 Serv-U。 

(1) 定位 到 新 添加 的 文件 目录 ， 找 到 可 执行 文件 Serv-U-Tray.exe， 双 击 该 程序 即 可 启动 本 地 FTP 
服务 端 ， 进 入 控制 管理 台 ， 首 先 需要 设置 上 传 。 其 中 管理 控制 台 的 运行 界面 如 图 4.11 所 示 。 

(2) 单 击 “ 域 详细 信息 ”选项 ， 进 入 域 创建 界面 。 如 果 为 首次 设置 域 信息 ， 则 需要 新 建 域 ， 如 
图 4.12 所 示 。 
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了 放 在 扫 二 的 用 户 罕 二- “内 所 有 有 内 户 权时 日 流 用 
和 Wndow3 册 户 半 组 人 
amod and dle LPP bost grouns, 


目录 
“为 峭 身 商 和 有 用 户 吕 香 巨 录 沪 问 权限 > 
， 晶 键 开间 护 庶 扒 间 弃 以 责 矶 问候 目录 以 4 二 ， 
限制 和 设置 
加 mn 
RTX 
A EE 


和 
《4 让 证 HTVL 设 置 * 
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图 4.11 管理 控制 台 的 运行 界面 图 4.12 单 击 “ 新 建 域 ”按钮 设置 域 信息 
(3) 进入 “ 域 向 导 ” 设 置 界面 。 需 要 进行 4 步 设 置 : @ 域名 称 ，@ FTP 协议 、 端 口 ; @ 可 用 
IP 地 址 ，@ 密码 加 密 方式 。 其 中 ， 设 置 FTP 端口 、 于 丙 和 下 13 和 图 4.14 所 示 。 


区 欢迎 使 用 Serv 城 向 导 。 本 向 导 档 帮助 人 在 文件 服务 器 上 Bi 寻 域 。 区 欢迎 使 用 Serv 域 向 导 。 本 向 将 帮 助 您 在 文件 服务 器 上 他 建城。 
可 以 合用 二 通过 各 种 协议 提供 对 文件 服务 器 的 访问 。 如 果 当前 许可 证 不 支持 时 Sevy 使用 电子 邮件 必 庆 用 息 ) 。 使 用 双 问 加 琵 来 保 
规 协 议 ， 风 j 些 协议 可 能 无 法 使 用 。 请 寺 择 域 应 访 使 用 的 协议 及 其 相应 交 庙 。 使 用 单 向 加 二 人 妇 任 “* 
Fp 和 ElotssunLs 本 可 使 用 服务 器 设置 “加 
局 全 单 向 加 密 《更 安全 ) 
国 mplct FTPS (SSUTLS) 990 € a 
团 使 用 SSH 的 SFTP 22 个 无 加 证 (不 推荐 ) 
Eh EE EE 
国 HTTPS (SSL 加 密 的 HTTP) 443 
固 区 许 用 户 恢复 密码 
E23| ES 
图 4.13 FTP 端口 设置 界面 图 4.14 ”FTP 密码 加 密 方式 设置 界面 


人 如 果 用 户 只 是 在 本 机 上 测试 ， 建 议 读者 不 要 更 改 参数 设置 。 
(4) 填写 域名 称 ， 然 后 单 击 “ 下 一 步 ”按钮 直至 设置 完成 。 


@ 
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(5) 完成 新 建 域 操作 后 ， 返 回 管理 控制 台 界 面 。 单 击 “ 添 加 新 用 户 ” 选 项 创建 新 用 户 。 单 击 “ 添 
加 ”按钮 进入 用 户 添加 界面 ， 在 该 界面 可 以 设置 个 人 用 户 信息 ， 即 FTP 服务 端 预 设 信息 。 选 择 “ 用 户 
信息 ”选项 卡 ， 在 该 页 面 中 填写 登录 ID、 全 名 、 密 码 以 及 选择 设置 根 目录 、 管 理 权限 、SSH 公共 密 钥 
路 径 、 账 户 类 型 、 说 明 等 信息 。 当 用 户 填 写 完毕 后 ， 单 击 “ 保 存 ” 按 钮 即 可 完成 用 户 添加 操作 。 其 中 ， 
添加 用 户 的 操作 界面 如 图 4.15 所 示 。 


| 用 户 信息 | 群 组 | 欢迎 消息 | 访问 | 目录 访问 ] 虚拟 路 径 [限制 和 设置 | 事件 | 
& 用 户 账户 信息 指定 授 子 该 账户 的 登录 攒 证 和 权限 。 


登录 ID: 全 名: 

Mr Mrsoft 

Ei 根 上 

<< 已 加 要 >> 匡 | CeyAppserwmw > >- 

管理 权限 : A 

无 权限 ~ 国 证 Frp 根 目录 
名 建 寺 衣 对 

三 户 类 型: 

永久 帐户 = 

点 认 Web 客户 庆 


提示 用 户 使 用 何 种 过户 ; -|“ 国 锐 定 用 户 至 根 司 江 
子 好 件 地 址 ” 国 


电子 邮件 地 址 : 登录 

口 用 户 必须 在 下 一 次 登录 时 更 下 密码 
回 启用 账户 
说 明 : 


| -| [加 可 用 性 ..… 


[ 保 友 | 吏 消 | [一 | 


图 4.15 添加 用 户 的 操作 界面 


/ 
jw 
便 地 访问 PHP 服务 器 目录 ， 从 而 使 FTP 管理 系统 发 挥 其 更 大 的 作用 和 优势 。 笔 者 在 设置 Serv-U 服 
务 端 用 户 时 ， 将 FTP 访问 的 根 目录 设置 成 Appserv 集成 环境 所 在 的 目录 。 


至 此 ，FTP 服务 器 端的 架设 和 参数 设置 介绍 完毕 。 如 读者 需要 对 其 他 参数 进行 设置 ， 建 议 参考 官 
方 提供 的 有 关 资料 。 由 于 篇 幅 所 限 ， 其 他 的 设置 参数 不 再 一 一 列举 。 


4.3 用 户 登 录 


4.3.1 用 户 登 录 功 能 概述 


用 户 登录 功能 是 FTP 服务 器 端 对 用 户 输入 的 IP 地 址 、 用 户 名 等 FTP 连接 信息 进行 验证 。 与 普通 
登录 验证 方式 不 同 的 是 , 用 户 在 登录 页 面 提交 的 验证 信息 需要 与 指定 服务 器 的 预 设 登录 信息 进行 比 对 ， 
如 果 用 户 输入 的 连接 信息 与 预 设 信息 匹配 , 则 可 以 顺利 连接 FTP 服务 器 , 否则 将 不 能 连接 FTP 服务 器 。 
这 样 ， 就 可 以 使 用 户 安全 地 连接 FTP 服务 器 。 登 录 页 面 的 运行 结果 如 图 4.16 所 示 。 


PHP 典型 模块 开发 全 程 实录 


”FTP 恩 理 系统 


全 . 
TP 地 址 : [27.0.0.1 
用 记名 : hr 
密码 : Jeeeeee 
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图 4.16 登录 页 面 的 运行 结果 
用 户 登录 功能 实现 包含 index.php (登录 表单 页 ) 和 config.php( 系统 连接 公共 文件 ) 2 个 核心 文件 ， 
它们 用 来 完成 对 登录 用 户 的 验证 操作 。 
4.3.2 登录 页 面 设计 
登录 表单 页 (index.php) 的 结构 相对 简单 ， 主 要 包含 连接 FTP 服务 器 IP 地 址 的 文本 框 、 用 户 名 文 
本 框 、 密 码 文本 框 和 登录 按钮 。 登 录 页 面 中 使 用 的 主要 元 素 如 表 4.1 所 示 。 
表 4.1 登录 页 面 中 使 用 的 主要 元 素 


元 素 名 称 说 明 

Script "js/check.phpy 登录 信息 是 否 为 空 验证 脚本 
text id="ip"name="ip" 连接 服务 器 IP 地 址 

text 连接 用 户 名 

password 连接 用 户 密码 

submit “登录 ”按钮 


在 FTP 管理 系统 的 登录 页 中 ， 通 过 checkjs 脚本 中 的 chklogin0 函 数 对 提交 的 登录 信息 进行 验证 。 
在 chklogin0 函 数 中 ， 通 过 获取 id 值 为 login 的 表单 中 的 元 素来 判断 文本 框 是 否 为 空 。 如 果 表 单元 素 中 
的 文本 框 为 室 ， 则 将 焦点 定位 到 该 元 素 上 。 代 码 如 下 : 

function chklogin(X 


if(login.ip.value==""){ 


alert(" 请 填写 连接 IP");login.ip.focus();return false; /| 判断 连 接 的 IP 文本 框 是 否 为 空 


if(login.username.value==""){ 


alert(" 请 填写 连接 用 户 名 ");login.username.focus();return false;”// 判 断 连 接 的 用 户 名 是 否 为 空 


if(login.password.value=="™"){ 


alert(" 请 填写 连接 密码 ");login.password.focus();return false;  // 判 断 连接 的 密码 是 否 为 空 


} 
login.submit(); /提交 login 表单 
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4.3.3 系统 连接 公共 文件 


检测 FTP 登录 用 户 信息 的 操作 在 config.php 文件 中 完成 。 首 先 获 取 表 单 提 交 的 登录 数据 ， 并 且 将 
其 赋 给 指定 的 Session 变量 。 代 码 如 下 : 


<?php 

iflisset($_POST[Ilogin']))f // 判 断 是 否 以 POST 方式 获取 login 传递 参数 
$_SESSION[ip]= $_POST[ip]; // 获 取 验 证 IP 地 址 并 赋 给 Session 变量 
$_SESSION['user] = $_POST['username'; /| 获取 验证 用 户 名 并 赋 给 Session 变量 
$_SESSION['pwd'] = $_POST[password]; // 获 取 验 证 密码 并 赋 给 Session 变量 

} 

S$ip =$_SESSION['ip]; /传递 Session 变量 给 相应 变量 


$user =$_SESSION['user]; 

$pwd = $_SESSION[pwd']; 

然后 ， 将 获取 的 FTP 地 址 、 用 户 名 称 、 密 码 作为 连接 参数 ， 远 程 连接 IP 地 址 指向 的 服务 器 。 如 果 
连接 成 功 ， 则 显示 main.php 页 面 的 其 他 内 容 ， 如果 连 接 失 败 ， 则 返回 错误 信息 。 相 关 代码 如 下 : 


S$link = @ftp_connect($ip); // 设 置 远程 服务 器 连接 标识 
S$login = @ftp_login($link, $user, $pwd); /| 连接 远程 FTP 服务 器 
if(l$login){ 1/ 判断 用 户 是 否 登录 成 功 
echo ' 错 误 ，<a href="index.php"> 返 回 </a>'; // 否 则 返回 错误 提示 
exit(); 

?> 


SS 全 四 


B 

本 章 系统 将 config 文件 目录 下 的 config.php 文件 作为 FTP 连接 的 公共 文件 。 为 了 避免 重 
复 书写 FTP 连接 过 程 ， 在 实现 其 他 FTP 功能 时 ， 只 需 通 过 include 语句 调用 该 公共 文件 完成 与 FTP 
服务 器 的 连接 即 可 。 


4.4 根 目录 操作 


4.4.1 根 目录 功能 概述 


用 户 在 正常 登录 FTP 管理 系统 后 ， 可 以 遍历 用 户 在 FTP 服务 器 中 指向 的 系统 根 目录 。 用 户 可 以 
单 击 根 目录 进入 次 级 根 目录 ， 并 可 对 根 目 录 执 行 操作 。 进 入 FTP 管理 系统 的 根 目录 运行 结果 如 图 4.17 
所 示 。 

其 中 ， 根 目录 的 主要 功能 包括 查看 根 目录 名 称 、 删 除 目 录 、 创 建文 件 目录 、 上 传 文件 到 根 目 录 等 。 
其 主要 是 在 main.php〈 系 统管 理 页 ) 、mkdirphp“〔〈 创 建文 件 目录 页 ) 、rmkdirphp〔〈 删 除 文件 处 理 页 ) 


等 文件 中 实现 。 
@ 
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ftp 服 务 器 的 拘 作 系统 是 WIX 。 ”当前 时 间 : 2011 年 05 月 09 日 » 创建 目录 上 忧 文 御 


4.17 FTP 管理 系统 的 根 目录 运行 结果 


4.4.2 遍历 文件 根 目录 


遍历 文件 根 目 录 ， 即 用 户 进入 FTP 系统 管理 页 面 (main.php) ， 应 用 ftp_rawlistO) 函 数 结合 foreach 
语句 输出 根 目录 中 的 所 有 文件 目录 名 称 。 用 户 单 击 文件 目录 名 称 的 超 链 接 进入 次 级 目录 ;， 如 果 用 户 想 
返回 根 目录 或 上 级 目录 ， 单 击 “ 返 回 上 级 目录 ” 超 链 接 ， 即 可 返回 父 级 目录 。 


1. 遍历 根 目录 名 称 
应 用 ftp_pwdO0、ftp_rawlistO 函 数 获取 路 径 名称 、 目 录 列 表 以 及 目录 下 的 文件 详细 信息 。 代 码 如 下 : 


<?php 


$dirname = ftp_pwd($link); // 返 回 当前 目录 名 

if($dirname != 1X{ /| 判断 当前 目录 中 是 否 存在 “/” 
S$dirname .= 几 // 不 存在 则 将 路 径 名 称 赋值 为 “/” 
} 

S$flist = ftp_rawlist($link, $dimame); 1/ 返回 当前 目录 下 的 详细 列表 
$sdir = array(); // 更 改变 量 $sdir 的 数据 类 型 
foreach($flist as $valueX{ // 循 环 输出 目录 详细 列表 

$array = explode(' ', rtrim($value)); /将 去 除 右边 空白 字符 的 元 素 ， 以 空格 作为 分 割 符 分 隔 
$alen = count($array); // 获 取 数 组 元 素 个 数 

$p = substr($array[0], 0, 1); /截取 返回 数组 长 度 

if($p == 'd'X{ 


if($array[$alen-1] != '.' && $array[$alen-1] (= "..'){ 
$sdirl] = $array[$alen-1]; 
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将 获取 的 当前 目录 名 称 、 目 录 列 表 以 及 文件 详细 列表 ， 在 显示 页 中 应 用 foreach 语句 循环 输出 。 代 
码 如 下 : 


<div class="mainleft"> 

<table border="0" cellpadding="0" cellspacing="0"> 

<?php 

// 循 环 输出 根 目录 的 目录 名 称 列表 

foreach($sdir as $value) 

echo'<tr> 

<td height="10" width="30">&nbsp;&nbsp;</td> 

<td height="25" width="75" align="center" class="STYLE1"> 

<a href="main.php?dir='.$dirname.$value.">'.substr($value,0,12).'</a> 
</td> 

<td width=50><a href="rmkdir.php?dir='".$dimame.$value."> 删 除 </a></td> 
<td height="5" width="30">&nbsp;&nbsp;</td> 

</tr>", 

?> 

</table> 

</div> 


2. 返回 上 级 目录 


返回 上 级 目录 是 通过 单 击 “ 返 回 上 级 目录 ” 超 链接 来 实现 的 。 其 实现 原理 是 ， 根 据 URL 地 址 中 传 
递 的 目录 路 径 dir， 应 用 ftp_chdir0 函 数 改变 当前 路 径 为 指定 路 径 。 相 关 代码 如 下 : 


<?php 

ifisset($_GET[dir])) /获得 以 GET 方式 传递 的 URL 参数 
ftp_chdir($link,$_GET[dir]); /改变 当前 路 径 为 指定 路 径 

} 

iflisset($_GET[up])X 1/ 判断 用 户 是 否 单 击 “ 返 回 上 级 目录 ” 超 链 接 
ftp_cdup($link); /改变 上 级 目录 为 指定 路 径 

if(isset($_GET[ down'])){ /判断 用 户 是 否 单 击 文件 目录 名 称 
ftp_get($link, $_GET[down],$_GET[down], FTP_BINARY); /进入 下 级 文件 目录 

} 

学 


4.4.3 删除 文件 目录 


在 系统 的 管理 页 面 中 ， 每 一 条 显示 文件 目录 名 称 后 面 都 有 一 个 “删除 ” 超 链接 。 单 击 “ 删 除 ” 超 
链接 ， 将 文件 目录 名 称 作为 参数 传递 到 rmkdir.php 文件 ， 完 成 文件 目录 的 删除 操作 。 相 关 代码 如 下 : 


<?php 
session_start(); // 开 启 全 局 会 话 
S$ip =$_SESSION['ip]; /将 Session 变量 中 的 数据 赋值 给 对 应 变量 


$user = $_ SESSION[user]; 
$pwd = $_SESSION[pwd']: 
S$link = @ftp_connect($ip); /| 连接 到 指定 FTP 地 址 


@ 
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$login = @ftp_ login($link, $user, $pwd); /连接 服务 器 

S$file = $_GET[dir].$_GET['file"; 1/ 设置 删除 文件 位 置 

if(ftp_delete($link, $file)X{ 1/ 判断 文件 是 否 被 删除 
echo '<script>alert(" 删 除 文件 成 功 ! ");history.back();</script>'; // 删 除 成 功 提示 

}else{ 
echo '<script>alert(" 删 除 文件 失败 !");history.back();</script>'; // 删 除 失败 提示 

} 

ftp_close($link); /关闭 FTP 连接 

?> 


4.4.4 在 根 目录 中 创建 新 目录 


单 击 系统 管理 页 面 中 的 “创建 目录 ” 超 链接 ， 跳 转 到 mkdir.php 页 面 ， 填 写 要 在 根 目 录 中 创建 的 新 
目录 名 称 ， 最 终 将 数据 提交 到 mkdir_chk.php 文件 中 ， 完 成 新 目录 的 创建 。 
在 mkdir.php 文件 中 ， 完 成 目录 名 称 的 添加 ， 其 代码 如 下 : 


<link href="css/mkdir.css” type="text/css" rel="stylesheet" /> 

<script src="js/check.js"></script> 

<form id="mkdir action="mkdir_chk.php?dir=" method="post" onSubmit="retum chkname();"> 
<input id="dimame" class="inpt1" type="text" name="filename" /> 

<input type="submit" value=" 提 交 " /> 

<input type="reset" value=" 重 置 " /> 

</form> 


在 mkdir_chk.php 文件 中 , 获取 表单 提交 的 目录 名 称 , 应 用 ftp_mkdir0 函 数 在 FTP 服务 器 根 目录 中 
创建 新 文件 目录 。 代 码 如 下 : 


<?php 

session_start(); 

$ip = $_SESSION['ip'; // 将 Session 变量 中 的 数据 赋值 给 对 应 变量 
$user = $_ SESSION[user]; 

$pwd = $_ SESSION[pwd]; 


if(lempty($_GET["dir"])){ // 判 断 是 否 通过 GET 方式 获取 文件 目录 路 径 
$path=$_GET["dir"]; // 将 路 径 名 称 赋值 给 对 应 变量 
}else{ 
$path=""; // 否 则 为 根 目录 
} 
$dirname=$path.$_POST['filename'"]; /指定 创建 文件 路 径 
$link = @ftp_connect($ip); /连接 指定 FTP 地 址 
S$login = @ftp_login($link, $user, $pwd); /连接 FTP 服务 器 
$result = ftp_mkdir($link, $dirname); /创建 文件 目录 
if($result){ // 判 断 文件 目录 是 否 被 创建 
echo '<script>alert(" 目 录 '.$dirname.\n 创建 成 功 ! ");window.close();</script>; ”// 提 示 创 建成 功 
}else{ 
echo '<script>alert(" 目 录 '.$dirname.\n 创建 成 功 !");history.back();</script>"; /提示 创建 失败 
} 
ftp_close($link); /| 关闭 FTP 连接 
?> 


@ 
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4.4.5 ”上 传 文件 到 根 目录 下 


在 系统 管理 页 中 ， 单 击 “ 上 传 文件 ” 超 链接 ， 跳 转 到 upfile.php 页 面 ， 选 择 要 上 传 的 本 地 文件 ， 将 
数据 提交 到 upfile_chk.php 文件 中 ， 实 现 将 本 地 文件 上 传 到 服务 器 指定 的 文件 夹 下 。 
在 upfile.php 文件 中 创建 表单 并 添加 文件 域 ， 将 本 地 文件 提交 到 upfilechk.php 文件 中 。 代 码 如 下 : 


<form action="upfile_chk.php" method="post" enctype="multipart/form-data"> 
<input class="inpt1” type="file" name="file" /> 

<input class="inpt1" type="submit" value=" 提 交 " /> 

</form> 


在 upfile_chk.php 文件 中 获取 表单 中 提交 的 本 地 文件 信息 ， 应 用 ftp_put0 函 数 将 文件 上 传 到 服务 器 
的 根 目录 中 。 相 关 代 码 如 下 : 


<?php 

session_start(); // 开 启 全 局 会 话 
include_once("config/config.php"); // 加 载 公共 连接 文件 
S$up=$_FILES["file"]["tmp_name"]; 1/ 获取 上 传 文件 
$file=$_FILES["file"]["name"]; /定义 上 传 文件 名 称 
if(ftp_put(S$link, $file, $up,FTP_BINARY)X // 上 传 文件 到 服务 器 根 目录 中 
echo '<script>alert(" 上 传 .$file.\n 成 功 ! ");window.close();</script>"; /提示 上 传 成 功 
}else{ 

echo '<script>alert(" 上 传 '.$file.\n 失败 ! ");history.back();</script>"; // 提 示 上 传 失败 

} 

ftp_close($link); /关闭 FTP 连接 

?> 


4.5 子 目 录 操 作 


4.5.1 子 目录 功能 概述 


用 户 单 击 根 目录 文件 会 进入 次 级 目录 ， 并 在 主 窗 体 显 示 该 文件 目录 中 的 子 目录 和 目录 下 的 所 有 文 
件 。 子 目录 中 部 分 操作 和 根 目录 操作 大 同 小 异 。 但 是 用 户 进入 子 目 录 后 ， 不 但 可 以 操作 子 文件 目录 ， 
还 可 以 对 文件 目录 下 的 文件 执行 多 种 操作 。 进 入 子 目录 的 运行 结果 如 图 4.18 所 示 。 

用 户 对 子 目 录 执 行 的 操作 包括 查看 子 目 录 文件 ， 删 除 子 目录 ， 在 子 目 录 中 创建 新 文件 目录 ， 上 传 
文件 到 当前 子 目录 ， 更 改 子 目录 下 的 文件 名 称 ， 移 动 文件 到 其 他 目录 中 ， 以 及 下 载 远 程 文 件 到 本 地 计 
算 机 中 。 对 子 目 录 的 操作 ， 主 要 是 在 main.php〈 系 统管 理 页 ) 、removefile.php〈 删 除 文件 处 理 页 ) 、 
movefile.php 移 动 文件 处 理 页 ) 等 文件 中 完成 的 。 


/ 
下 全 由 于 目录 的 创建 和 上 传 文件 等 功能 与 根 目录 操作 大 体 相同 ， 所 以 在 后 面 章节 中 不 再 进行 
讲解 。 下 面 只 对 子 目 录 中 特有 的 一 些 功能 进行 讲解 。 
© 
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图 4.18 子 目录 的 运行 结果 
4.5.2 浏览 子 目录 下 的 文件 


用 户 登录 FTP 管理 系统 后 ， 单 击 任意 根 目录 进入 次 级 目录 ， 在 主 窗 体 同时 会 显示 子 目录 中 包含 的 
所 有 文件 的 相关 信息 。 其 中 ， 显 示 内 容 包 括 文件 名 称 、 文 件 大 小 、 最 后 修改 时 间 等 。 另 外 ， 用 户 也 可 
以 对 该 目录 下 的 文件 执行 移动 、 更 改名 称 等 操作 。 在 main.php 文件 中 ， 人 遍历 子 目录 中 所 有 文件 相关 信 
息 的 关键 代码 如 下 : 


<?php 
$dirname = ftp_pwd($link); 1/ 返回 当前 目录 名 
if($dirname != YI // 判 断 当前 目录 中 是 否 存在 “/” 
$dirname .= 几 // 不 存在 则 将 路 径 名 称 赋值 为 “/” 
} 
$sfile = ftp_nlist($link, $dirname); // 返 回 当前 根 目 录 列 表 
if(l$sfile)f // 判 断根 目录 下 是 否 为 空 

$sfile = array(); // 更 改变 量 $sfile 的 数据 类 型 
} 
S$flist = ftp_rawlist($link, $dimame); // 返 回 当前 目录 下 的 详细 列表 
是 > 


通过 foreach 语句 循环 输出 子 目录 中 的 详细 内 容 。 关 键 代 码 如 下 : 


<table border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor="#BAE8BC"> 
<tr> 
<td height="35" colspan="6" bgcolor="#FFFFFF"><div style=" margin:auto 20px;text-align:right;" > 
<input type="button" value="" class="btn3" onclick="javascript:Wopen=open('path_mkdir.php? 
dir=<?php echo $dirname; ?>",",'height=220,width=350,scrollbars=no');"> 
<input type="button" class="btn4" onclick="javascript:Wopen=open('‘path_upload.php? 


@ 
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dir=<?php echo $dirname; ?>',",height=220,width=350,scrollbars=no);"></div> 
</td> 
</tr> 
<tr> 

<td height="28" align="center" bgcolor="#FFFFFF"> 文 件 名 称 </td> 
<td align="center" bgcolor="#FFFFFF"> 文 件 大 小 </td> 
<td align="center" bgcolor="#FFFFFF"> 最 后 修改 时 间 </td> 

<td align="center" bgcolor="#FFFFFF"> 更 改名 称 </td> 
<td align="center" bgcolor="#FFFFFF"> 删 除 文件 </td> 

<td align="center" bgcolor="#FFFFFF"> 移 动 文件 </td> 
</tr> 
<?php 
foreach($sfile as $value){ // 循 环 输出 文件 列表 
?> 
<tr> 
<td width="250" height="30" align="center bgcolor="#FFFFFF"> 
<a href="#" onclick="javascript:Wopen=open(download.php?dir=<?php echo $dimame; ?> 
&filename=<?php echo $value;?>',",'height=220,width=350,scrollbars=no');"><?php echo $value; ?></a> 
</td> 
<td width="70" height="30" align="center" bgcolor="#FFFFFF"><?php echo round(ftp_size($link,$value)/1024, 
2);?> K 
</td> 

<td width="200" height="30" align="center" bgcolor="#FFFFFF"> 
<?php echo date(Y-m-d Hiis'ftp_mdtm($link,$value)); ?> 
</td> 
<td width="60" height="30" align="center" bgcolor="#FFFFFF"> 
<a href="#" onclick="javascript:Wopen=open(edit.php?file=<?php echo $value; ?> 
&dir=<?php echo $dirname; ?>',",'height=220,width=350,scrollbars=no'); "> 更 改 </a> 
</td> 

<td width="60" height="30" align="center" bgcolor="#FFFFFF"> 

<a href="removefile.php?file=<?php echo $value; ?>&dir=<?php echo $dimame;?>"> 删 除 </a> 
</td> 
<td width="60" height="30" align="center" bgcolor="#FFFFFF"> 
<a href="#" onclick="javascript:Wopen=open('movefile.php?dir=<?php echo $dirname; ?> 
&filename=<?php echo $value;?>',",'height=220,width=350,scrollbars=no');"> 移 动 </a></td> 
</tr> 
<?php 
} //foreach 语句 结束 
Th 
</table> 


4.5.3 ”远程 文件 下 载 


作为 一 款 FTP 管理 工具 ， 下 载 远 程 服务 器 中 的 文件 这 一 功能 必 不 可 少 , 与 普通 Web 程序 下 载 功能 
不 同 ，FTP 服务 器 上 的 文件 并 不 能 直接 通过 单 击 超 链 接 形式 下 载 。 因 为 FTP 服务 器 指定 的 目录 不 一 定 
是 Web 服务 器 或 提供 服务 的 环境 目录 ， 所 以 这 里 需要 通过 ftp_get() 函 数 来 执行 远程 文件 下 载 操作 。 

单 击 “ 文 件 名 称 ” 超 链接 ,进入 download.php 文件 下 载 页 面 , 填写 正确 的 本 地 下 载 路 径 , 单 击 “ 下 


@ 
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载 ” 按 钮 ， 将 数据 提交 到 download.php 文件 中 ， 将 远程 服务 器 中 的 文件 下 载 到 本 地 。download.php 文 
件 的 代码 如 下 : 


<?php 
session_start(); /开启 全 局 会 话 
include_once("config/config.php"); // 加 载 公共 连接 文件 
if(lempty($_POST["savepath"])X{ /| 判断 是 否 存在 下 载 路 径 
$absopath=$_POST["savepath"]; /获取 下 载 路 径 
$dir=$_GET["dir]; /获取 下 载 的 文件 路 径 
$dwname=$_GET["filename"]; /获取 下 载 的 文件 名 称 
$file=$dir. $dwname; // 定 义 下 载 文 件 
$down=$absopath.$dwname; /定义 下 载 路 径 
if(ftp_get($link,$down,$file,FTP_BINARY)){ /| 判断 是 否 从 FTP 服务 器 中 下 载 文件 

echo '<script>alert(" 文 件 '.$file."\n 下 载 成 功 ! ");window.close();</script>"; 
}else{ 

echo '<script>alert(" 文 件 '. $file.\n 下 载 !");history.back();</script>"; 

} 

)_close( $link); /| 关闭 FTP 连接 
}else{ // 如 果 下 载 路 径 不 存在 ， 则 向 用 户 展示 下 载 提 示 框 
?> 
<script src="js/check.js"></script> 
< 省 略 部 分 代码 一 > 


<form id="download" action="#" method="post" onsubmit="return chkdown();"> 
<input id="down" class="ipt1" type="text" name="savepath" /> 
<p><span style=" color:#FF0000;">* 请 填写 有 效 路 径 (如 C:\LeonWAppserw)</span></p> 
<input class="btn" type="submit" value=" 下 载 " />&nbsp;&nbsp;<input class="btn" type="reset" value=" 重 置 " /> 
</form> 
<?php 
/| 结束 选择 判断 


?> 


V4 
后 曙 (1) 远程 文件 下 载 操作 中 ， 路 径 输入 的 form 表单 与 远程 文件 的 下 载 操作 均 存储 于 
down.php 文件 中 。 
(2 ) 远 程 文件 下 载 功能 中 输入 的 下 载 路 径 要 按照 路 径 规范 书写 。 如 图 4.19 所 示 定 义 的 就 
是 一 个 规范 的 文件 下 载 路 径 。 


C: \TEMP\ 


# 请 填写 有 效 路 径 (如 C:\Leon\Appserv\) 


下 载 | | 重 秆 


图 4.19 定义 规范 的 远程 文件 下 载 路 径 


@. 
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4.5.4 文件 移动 


用 户 可 以 选 定 文件 并 将 其 作为 对 象 移动 到 FTP 的 其 他 目录 中 。 单 击 文件 详细 列表 中 的 “移动 ” 超 
链接 ， 进 入 移动 路 径 的 填写 页 面 movefile.php 中 ， 添 加 完整 的 移动 路 径 ， 单 击 “移动 ”按钮 ， 将 数据 
提交 到 本 页 ， 完 成 文件 的 移动 操作 。movefile.php 文件 的 代码 如 下 : 


<?php 

session_start(); /开启 全 局 会 话 
include_once("config/config.php"); /| 加载 公共 连接 文件 
$filename=$_GET["dir"].$_GETI"filename"]; // 定 义 文件 路 径 
$fname=$_GET["filename"]; /| 获取 文 件 名 称 
if(empty($_POST["movepath"])X{ /| 判断 路径 是 否 存在 
?> 


<form id="movefile" action="#" method="post" onSubmit="return chkpath();"> 
将 <?php echo $filename; ?> 移动 到 : 


<p> 
<center> 

<input class="inpt1" id="path" type="text" name="movepath"> 
</center> 
<p> 
<center> 
<span class="notice">* 请 填写 正确 的 路 径 。" 如 /Appserv/"</span> 
</center> 
<p> 
<center> 
<input class="btn2" type="submit" value=" 移 动 文 件 "> 
</center> 
</form> 
<?php }else{ 
$newpath=$_POSTI"movepath"].$fname; // 如 果 存 在 ， 则 执行 移动 操作 
if(ftp_put($link, $newpath, $filename, FTP_BINARY)X /将 文件 移动 到 指定 文件 目录 
unlink($filename); /删除 原来 位 置 的 文件 

echo '<script>alert(" 移 动 '.$file.\n 到 '.$_POST[movepath].' 成 功 ! ");window.close();</script>'; 

}else{ /提示 移动 成 功 

echo '<script>alert(" 移 动 '.$file.\n 到 '.$_POST[movepath].' 失 败 ! ");history.back();</script>'; 
. // 提 示 移 动 失败 
ftp_close($link); /关闭 FTP 连接 
]}?> 


4.5.5 ”更改 文件 名 称 


在 FTP 管理 系统 中 ， 用 户 不 但 可 以 执行 访问 、 查 看 、 下 载 文件 等 操作 。 同 时 如 果 在 移动 文件 过 程 
中 ， 文 件 名 称 冲 突 可 能 会 导致 移动 失败 ， 所 以 开发 了 文件 名 称 更 改 的 功能 。 单 击 文件 详细 信息 列表 中 
指定 文件 后 的 “更 改 ” 超 链接 ， 跳 转 到 edit.php 页 面 中 ， 添 加 新 的 文件 名 称 ， 单 击 “ 更 改 ” 按钮 ， 将 数 
据 提交 到 本 页 ， 完 成 对 文件 名 称 的 更 改 操作 。edit.php 文件 的 关键 代码 如 下 : 
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<!-- 创 建 添加 文件 名 称 的 表单 --> 
<table> 
<!-- 省 略 部 分 代码 -> 
<td width="770" align="center"><table border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td align="center" class="STYLE1"> 
<form id="form1" name="form1" method="post" 
action="edit.php?dir=<?php echo $_GET['dir].'&file=.$_GET[file] ?>" onSubmit="retum chkamend();"> 
<p> 原 文件 名 为 : <?php echo $_GET[file]; ?></p> 
<p> 输 入 文件 名 : 
<input id="news" type="text" name="new" /> 
&nbsp; 
<input type="submit" name="Submit" value=" 更 改 " /> 
</p> 
<p> 注 意 : 要 输入 文件 的 全 名 ， 包 括 扩展 名 </p> 
</form> 
</td> 


获取 URL 参数 ， 应 用 ftp_rename() 函 数 更 改 文件 名 称 。 代 码 如 下 : 


<?php 

/获取 表单 提交 的 文件 名 称 ， 完 成 对 指定 文件 名 称 的 更 改 操作 

session_start(); /开启 全 局 会 话 
include_once("config/config.php"); 1/ 加载 公 共 连 接 文件 

$file =$_GET[dir].$_GET[file]; /| 获取 文 件 完整 路 径 

if($_POST){ /| 判断 是 否 存在 以 POST 方式 提交 的 数组 
$new = $_GET['dir].$_POST[new'; // 获 取 新 命名 文件 的 完整 路 径 

$result = ftp_rename($link, $file, $new); /| 更改 名称 

if($result) // 判 断 是 否 更 改 成 功 

{ 

echo '<script>alert(" 文 件 '$_GET['file].\n 改名 为 m'.$_POST[new]."n 成 功 ");window.close();</script>'; 
} // 提 示 更 改 成 功 

else{ 

echo '<script>alert(" 文 件 .$_GET[file].\n 改名 为 \n".$_POST[new"].\n 失败 ");history.back()";</script>'; 
} // 提 示 更 改 失 败 


we 
4 明 更 改 文件 名 称 时 必须 添加 扩展 名 ， 如 index.php 而 不 是 index。 如 果 更 改 时 未 添加 扩展 名 ， 
可 能 导致 程序 无 法 正常 运行 。 


4.6 技术 提炼 


4.6.1 FTP 文件 操作 技术 


对 FTP 文件 的 操作 ， 应 用 的 是 PHP 内 置 的 FTP 文件 传输 函数 库 中 的 函数 。 下 面 对 其 中 常用 的 函 


@ 
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数 进行 详细 讲解 。 

(1) ftp_connectO) 函 数 

该 函数 用 来 建立 一 个 新 的 FTP 连接 ， 如 果 成 功 ， 则 返回 一 个 连接 标识 ， 否 则 返回 false。 该 函数 的 
语法 如 下 : 


resource ftp_connect( string host [, int port [, int timeout]] ) 


host: 必要 参数 。FTP 服务 器 的 地 址 ， 后 面 不 应 以 斜 线 结尾 ， 前 面 也 不 需要 以 ftp:// 开 头 。 

port: 可 选 参 数 。FTP 服务 器 的 端口 ， 默 认 端 口 为 21。 

回 timeout: 可 选 参数 。 设 置 网 络 传输 的 超时 时 间 限 制 ， 默 认 值 为 90 秒 ， 适 用 于 PHP 4.2.0 及 以 
上 版 本 。 

下 面 示例 应 用 ftpp_connectO 函 数 改 变 FTP 服务 器 上 一 个 文件 的 属性 。 代 码 如 下 : 


<?php 
$ftp_server = "127.0.0.1"; 
$link = ftp_connect($ftp_server) or die(" 不 能 连接 FTP 服务 器 "); /建立 FTP 连接 
if(Slink)f 
echo "连接 FTP 服务 器 成 功 "; 


} 
ftp_close($link); /关闭 FTP 连接 
?> 
(2) ftp_login() 函 数 
利用 该 函数 可 以 登录 到 FTP 服务 器 ， 如 果 成 功 返 回 tmue， 和 否则 返回 false。 该 函数 的 语法 如 下 : 


bool ftp_login( resource ftp_stream, string username, string password ) 


ftp_strea: 必要 参数 。FTP 连接 的 标识 符 。 

username: 必要 参数 。FTP 服务 器 的 用 户 名 。 

password: 必要 参数 。FTP 服务 器 的 密码 。 

下 面 示例 应 用 ftp_login0) 函 数 登 录 到 FTP 服务 器 。 代 码 如 下 : 


<?php 
$link = ftp_connect("127.0.0.1"); 
S$login_result = ftp_login($link, "mr", "mrsoft"); /登录 FTP 服务 器 
if($login_result) { 
echo "登录 成 功 "; 
}else{ 


echo "登录 失败 "; 


} 
ftp_close($link); /关闭 FTP 连接 


?> 


(3) ftp_chdir0 函 数 
利用 该 函数 可 以 在 FTP 服务 器 上 改变 当前 的 目录 为 指定 目录 ， 成 功 返 回 tue， 和 否则 返回 false。 该 
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函数 的 语法 如 下 : 
bool ftp_chdir( resource ftp_stream, string directory ) 
应 用 该 函数 切换 当前 目录 为 指定 目录 的 代码 如 下 : 


<?php 

Slink = ftp_connect("127.0.0.1"); 
S$login_result = ftp_login($link, "mr", "mrsoft"); 
ftp_chdir($link, "myfile"); 

echo "当前 目录 为 "ftp_pwd($link); 
ftp_close($link); 

?> 


(4) ftp_mdtm() 函 数 


1/ 连接 指定 地 址 

// 连 接 FTP 服务 器 
/改变 目录 

// 输 出 当前 的 目录 名 
/关闭 FTP 连接 


该 函数 用 来 检查 指定 文件 的 最 后 修改 时 间 ， 并 以 UNIX 时 间 改 的 方式 返回 ， 如 果 发 生 错误 ， 或 文 
件 不 存在 ， 则 返回 -1。 某 些 FTP 服务 器 可 能 会 不 支持 这 个 特性 。 该 函数 的 语法 如 下 : 


int ftp_mdtm( resource ftp_stream, string remote file ) 


ftp_stream: 必要 参数 。 指 定 FTP 连接 的 标识 符 。 
remote_file: 必要 参数 。 设 置 FTP 服务 器 的 名 称 。 
应 用 该 函数 检查 文件 修改 时 间 的 代码 如 下 : 


<?php 
S$file = "myfile/mingri.php"; 
Slink = ftp_connect("127.0.0.1"); 
S$login_result = ftp_login($link, "mr", "mrsoft"); 
$result = ftp_mdtm($link, $file); 
if($result == -1) { 
echo "查询 失败 "; 
}else{ 
echo $file." 的 最 后 修改 时 间 : ".date('Y-m-d H:i:s', $result); 


} 
ftp_close($link); 
?> 


(5) ftp_nlist0) 函 数 


// 设 置 文件 名 称 

// 连 接 指 定 地 址 

// 连 接 FTP 服务 器 

// 检 查 指定 文件 的 最 后 修改 时 间 
// 判 断 查询 的 文件 是 否 存在 


// 输 出 最 后 修改 时 间 


// 关 闭 FTP 连接 


该 函数 用 来 返回 给 定 目 录 的 文件 列表 ， 如 果 成 功 ， 则 返回 给 定 目录 下 的 文件 名 组 成 的 数组 ， 否 则 


返回 false。 该 函数 的 语法 如 下 : 


array ftp_nlist ( resource ftp_stream, string directory ) 


应 用 ftp_nlist0 函 数 返 回 给 定 目 录 的 文件 列表 的 代码 如 下 : 


<?php 
S$link = ftp_connect("127.0.0.1"); 


// 连 接 指定 地 址 
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S$login_result = ftpp_login($link, "mr", "mrsoft ); // 连 接 FTP 服务 器 

$result = ftp_nlist($link, "myfile"); // 返 回 给 定 目录 的 文件 列表 
print_r($result); // 输 出 数组 结果 集 
ftp_close($link); /关闭 FTP 连接 

?> 


(6) ftp_rename() 函 数 
该 函数 将 FTP 服务 器 上 的 一 个 文件 或 目录 改名 ， 如 果 成 功 则 返回 tue， 和 否则 返回 包 lse。 该 函数 的 
语法 如 下 : 
bool ftp_rename( resource ftp_stream, string oldname, string newname ) 


ftp_stream: 必要 参数 。FTP 连接 的 标识 符 。 
oldname: 必要 参数 。 旧 的 名 称 。 
newname: 必要 参数 。 新 的 名 称 。 

应 用 该 函数 实现 文件 或 目录 改名 的 代码 如 下 : 


<?php 
Slink = ftp_connect("127.0.0.1"); 
$login_result = ftp_login($link, "mr", "mrsoft"); 


$rename1 = ftp_rename(S$link, "myfile", "mynewfile"); /将 myfile 文件 目录 重新 命名 
if($rename1)f // 判 断 目录 名 称 是 否 被 更 改 
echo "更 改 目录 名 成 功 "; // 判 断 文件 目录 名 称 是 否 被 更 改 
}else{ 
echo "更 改 目 录 名 失败 "; 
} 
$rename2 = ftp_rename(S$link, "mr.php", "mingri.php"); // 将 mr.php 文件 重新 命名 
if($rename2){ // 判 断 文件 名 称 是 否 被 更 改 
echo "更 改 文件 名 成 功 "; 
}else{ 
echo "更 改 文件 名 失败 "; 
} 
ftp_close($link); /关闭 FTP 连接 
?> 


(7) ftp_get() 函 数 
该 函数 用 来 下 载 FTP 服务 器 上 的 文件 ， 如 果 成 功 返回 tue， 和 否则 返回 false。 该 函数 的 语法 如 下 : 


bool ftp_get( resource ftp_stream, string local_file, string remote _file, int mode [, int resumepos] ) 
ftp_get0 函 数 的 参数 说 明 如 表 4.2 所 示 。 
表 4.2 ftp_get() 函 数 的 参数 说 明 


参数 说 了 明 
ftp_stream 必要 参数 。FTP 连接 的 标识 符 
local file 必要 参数 。 保 存 到 本 地 的 文件 名 
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续 表 
参数 说 了 明 
remote file 必要 参数 。FTP 服务 器 上 的 文件 名 称 
mode 必要 参数 。 传 送 模式 参数 ， 为 FTP_ASCII (文本 模式 ) 或 FTP_BINARY (二 进 制 模式 ) 中 的 一 个 
resumepos 可 选 参数 。 仅 适用 于 PHP 4.3.0 以 上 版 本 


应 用 该 函数 下 载 指定 文件 的 代码 如 下 : 


<?php 
$down = "mr.php"; // 设 置 保存 到 本 地 文件 名 
S$file = "myfile/mingri.php"; 
Slink = ftp_connect("127.0.0.1"); /建立 FTP 连接 
S$login_result = ftpp_login($link, "mr", "mrsoft"); 
if(ftp_get($link, $down, $file,FTP_BINARY)) { 1/ 判断 是 否 从 FTP 服务 器 中 下 载 文件 
echo "下 载 $file 成 功 "; 
}else{ 
echo "下 载 $file 失败 "; 
} 
ftp_close($link); /关闭 FTP 连接 


?> 


(8) ftp_put() 函 数 
该 函数 用 来 上 传 一 个 已 经 打开 的 文件 到 FTP 服务 器 ， 如 果 上 传 成 功 ， 返 回 tue， 和 否则 返回 false。 
该 函数 的 语法 如 下 : 


bool ftp_put( resource ftp_stream, string remote _file, resource handle, int mode [, int startpos] ) 


ftp_put() 函 数 的 参数 说 明 如 表 4.3 所 示 。 
表 4.3 ftp_put() 函 数 的 参数 说 明 


。FTP 连接 的 标识 符 
remote file 必要 参数 。 本 地 已 经 打开 文件 的 句柄 
handle | 必要 参数 。FTP 服务 器 上 的 文件 名 称 
mode 。 传 送 模式 参数 ， 为 FTP_ASCII (文本 模式 ) 或 FTP_BINARY (二 进 制 模式 ) 中 的 一 个 


。 仅 适用 于 PHP 4.3.0 以 上 版 本 
应 用 该 函数 上 传 指定 文件 的 代码 如 下 : 


<?php 
$up = "soft.php"; /本 地 上 传 文件 名 
S$file = "myfile/mrsoft.php"; /指定 服务 器 中 的 目录 名 称 


S$link = ftp_connect("127.0.0.1"); 

S$login_result = ftp_login($link, "mr", "mrsoft"); 

if(ftp_put($link, $file, $up,FTP_BINARY)) { /| 判断 是 否 上 传 文件 到 指定 目录 中 
echo "上 传 $file 成 功 "; 


&. 
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}else{ 


echo "上 传 $file 失败 "; 


} 
ftp_close($link); /| 关闭 FTP 连接 
?> 


此 外 , 本 FTP 管理 系统 中 还 存在 其 他 的 FTP 函数 , 由 于 这 些 函 数 并 非 本 章 重点 , 所 以 不 再 一 一 介绍 。 
4.6.2 ”循环 输出 技术 


所 有 程序 设计 中 都 包含 循环 控制 语句 ， 用 于 反复 多 次 处 理 问 题 。 在 FTP 管理 系统 中 ， 遍 历 指定 文 
件 目录 下 的 内 容 ， 应 用 的 是 ftp_nlist0 函 数 ， 其 返回 值 为 数组 ， 所 以 选择 foreach 语句 循环 输出 文件 目录 
下 的 内 容 。foreach 语句 的 特点 就 是 操作 数组 ， 其 语法 如 下 : 


foreach (array_expression as $value) 
statement 


foreach (array_expression as $key => $value) 
statement 


其 中 ， 第 一 种 格式 遍历 给 定 的 array_expression 数组 。 每 次 循环 中 ， 当 前 单元 的 值 被 赋 给 $value， 
并 且 数 组 内 部 的 指针 向 前 移 一 步 〈 因 此 下 一 次 循环 中 将 会 得 到 下 一 个 单元 〉。 

第 二 种 格式 的 实现 功能 与 第 一 种 格式 相同 ， 只 是 在 输出 上 略 有 不 同 ， 即 元 素 循环 输出 时 ，S$key 被 
作为 指定 元 素 的 键 名 一 同 输出 。 


人 注意 foreach 语句 不 支持 “(@” 来 禁止 错误 信息 。 
例如 ， 在 本 系统 中 应 用 foreach 语句 来 循环 输出 子 目 录 中 文件 相关 信息 的 代码 如 下 : 


<?php 

$dirname = ftp_pwd($link); 

if($dirname != MX 

$dirname .= "/"; $flist = ftp_rawlist($link, $dimame); 
$sdir = array(); 

foreach($flist as $value}{ 

$array = explode( ", rtrim($value)); 

$alen = count($array); 

$p = substr($array[0], 0, 1); 

if($p == 'd'X( 

if($array[$alen-1] != … && $array[$alen-1] !=".."}{ 
$sdi = $array[$alen-1]; 

} 

} 


} foreach($sdir as $value) 
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echo'<tr> 
<td height="10" width="30">&nbsp;&nbsp;</td> 

<td height="25" width="75" align="center" class="STYLE1"> 

<a href="main.php?dir='.$dirname.$value.">'.substr($value,0,12).'</a> </td> 
<td width=50><a href="rmkdir.php?dir=".$dirname.$value."> 删 除 </a></td> 
<td height="5" width="30">&nbsp;&nbsp;</td> 

</tr>", 

?> 


4.6.3 应 用 CSS 设置 屏幕 滚动 条 技术 


FTP 管理 系统 的 结构 和 一 般 连 接 数据 库 的 Web 程序 不 同 ， 其 在 页 面 返回 的 结果 依据 实际 目录 中 的 
文件 ， 而 一 般 连 接 数据 库 的 Web 程序 则 是 通过 连接 数据 库 并 应 用 数据 库 查 询 语句 返回 查询 结果 。 连 接 
数据 库 的 Web 程序 可 以 通过 分 页 来 实现 数据 的 分 段 显 示 , 而 FTP 管理 系统 则 很 难 实现 这 样 的 分 段 显示 ， 
面 对 文 件 目 录 中 数目 很 多 的 目录 ， 返 回 的 文件 结果 将 会 撑 开 整个 屏幕 。 所 以 ， 这 里 应 用 CSS 相关 技术 
设置 屏幕 的 深 动 条 ， 实 现 文件 目录 下 数据 的 深 动 输出 ， 即 当 数 目 大 于 所 给 边框 长 度 时 ， 系 统 的 返回 结 

自动 判断 并 设置 深 动 条 的 CSS 的 关键 语法 如 下 : 


overflow:auto; // 设 置 自动 判断 并 设置 滚动 条 
另外 ， 也 可 以 设置 横向 深 动 条 和 纵向 深 动 条 。 语 法 格式 如 下 : 
overflow-x:auto; /设置 横向 自动 判断 并 设置 滚动 条 
overflow-y:hidden; // 设 置 纵向 自动 判断 并 隐藏 滚动 条 


自动 设置 滚动 条 的 一 般 过 程 如 下 : 
(1) 设置 div 布局 层 ， 并 为 该 层 设置 高 度 。 相 关 代码 如 下 : 
<dvoass="content> 


<!-HTML 代码 部 分 一 > 
</div> 


(2) 在 <head></head> 标 签 中 ， 添 加 <style></style> 属 性 标签 ， 并 设置 class 为 content 的 CSS 样式 
表 为 自动 调节 高 度 和 宽度 。 相 关 代码 如 下 : 


<head> 

<style type="text/css"> 
content{overflow:auto;height:200px;} 
</style> 

<head> 


这 样 ， 当 类 为 content 的 div 容器 中 内 容 高 度 超过 200px 值 时 ， 在 div 的 边框 会 自动 显示 滚动 条 。 
用 户 可 以 通过 拖 动 深 动 条 来 查看 类 为 content 的 div 中 所 显示 的 全 部 内 容 。 其 运行 效果 如 图 4.20 所 示 。 
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着 去 
明 目 FTP 管 理 系统 
tp 服务 器 的 办 作 系统 是 UK 当前 时 间 : 2011 年 05 朋 18 日 六 Ce [Te 
srt 滚动 条 的 添加 = 
adodb 最 除 
删除 创建 目录 | 上 待 文件 
ony 文件 大 小 最 后 外 MA 时 间 更 PS 名 称 “遇险 文件 “移动 文 恒 
册 出 除 
Ge asx 2011-02-18 11-34:08 Wy 了 
3 最 除 275k 2011-02-14 17:00:05 到 
estfiles 。 县 除 1 后天 2011-00-18 11:39:2 EE 
file 抽 除 
到 13K 2011-02-16 16:14:05 Wh MW 
sadepart shp ox 20t1-02-18 11:40:29 WM 
adadepart_chk php 108K 2011-02-16 11:25:01 a 删除 移动 -| 


图 4.20 应 用 CSS 样式 设置 滚动 条 
4.7 本 章 小 结 


本 章 通过 一 个 完整 的 FTP 管理 系统 使 读者 详细 了 解 使 用 FTP 进行 文件 上 传 与 下 载 的 操作 方法 。 通 
过 Serv-U 软件 设置 一 台 FTP 服务 器 , 用 户 或 其 他 使 用 者 就 能 够 通过 互联 网 或 者 同一 网 络 上 的 任何 一 台 
计算 机 与 FTP 服务 器 连接 ， 执 行文 件 或 目录 的 复制 、 移 动 、 创 建 和 删除 等 操作 。 


AS < 
于 
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上 网 浏览 网 页 时 , 经 常会 遇 到 打开 网 页 后 弹出 一 个 小 窗口 的 情况 ， 
如 今 很 多 大 企业 门户 网 站 ， 都 通过 这 种 弹出 式 窗口 发 布 一 些 公 告 或 者 
假日 问候 等 。 本 章 将 开发 一 个 假日 公告 管理 模块 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 


Ld 
Led 
ed 
ed 
ed 
Ld 


数据 库 的 连接 与 数据 操作 
通过 起 链接 传递 值 

单 文件 上 传 

多 图 片上 传 

读 取 指 定 目录 中 的 目录 及 文件 
使 用 jQuery 控制 弹出 公告 信息 
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5.1 假日 公告 管理 模块 概述 


假日 公告 管理 模块 实现 发 布 及 管理 假日 公告 信息 的 功能 ， 当 要 发 布 假日 公告 时 ， 在 假日 公告 发 布 
模块 中 上 传 公告 文件 及 图 片 即 可 ， 让 用 户 及 时 了 解 公司 的 工作 时 间 ， 并 且 可 以 第 一 时 间 为 用 户 送 上 假 
日 祝福 。 假 期 过 后 ， 可 以 通过 假日 公告 取消 模块 取消 发 布 的 假日 公告 ， 然 后 通过 假日 公告 管理 模块 将 
公告 信息 删除 。 


5.1.1 功能 结构 
假日 公告 管理 模块 的 功能 结构 如 图 5.1 所 示 。 


假日 公告 发 布 


堆 半 DID 高 机 | 
a | 
齐 j4Dly>ZD 剖 滨 守 


图 5.1 假日 公告 管理 模块 的 功能 结构 


5.1.2 程序 预览 


为 了 让 读者 更 好 地 了 解 假日 公告 管理 模块 的 功能 ， 下 面 列 出 假日 公告 管理 模块 中 一 些 功能 的 运行 
效果 ， 供 广大 读者 参考 。 
运行 网 站 ， 弹 出 公告 信息 页 面 ， 效 果 如 图 5.2 所 示 。 


苯 歼 的 读者 朋友 们 : 

由 于 新 春 佳节 假日 临近 ， 本 公司 于 加 11 年 1 月 31 日 到 
2011 年 ?月 8 日 放假 , 所 以 在 此 期 间 暂 停 服务 , 请 将 您 的 问 
是 发 到 技术 论坛 或 邮箱 中 ， 上 班 后 我 们 会 第 一 时 间 回 复 ， 
给 您 带 来 的 不 便 ， 请 谅解 ! 


图 5.2 ”公告 信息 页 面 效果 
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假日 公告 管理 模块 的 主页 面 运 行 效果 如 图 5.3 所 示 。 
登录 页 面 运行 效果 如 图 5.4 所 示 。 


eS A 


图 5.4 登录 页 面 运 行 效果 


导 
图 5.5 假日 信息 发 布 页 面 运行 效果 


面 运行 


效果 如 图 5.7 所 示 。 
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5.2 关键 技术 


5.2.1 数据库 的 连接 与 数据 操作 


1. 数据 库 的 连接 


(1) 要 操作 MySQL 数据 库 ， 首 先 必须 与 MySQL 服务 器 建立 连接 。 连 接 MySQL 服务 器 应 用 的 
是 mysql_connect0 函 数 ， 其 语法 如 下 : 


mysql_connect(hostname',username', password'); 
< 一 一 


MySQL 服务 器 的 用 户 密码 
登录 MySQL 数据 库 服务 器 的 用 户 名 


MySQL 服务 器 的 主机 名 (或 IP)， 如 果 省 略 端口 号 ， 默 认为 3306 


该 函数 的 返回 值 用 于 表示 这 个 数据 库 连 接 。 如 果 连 接 成 功 , 则 函数 返回 一 个 资源 , 为 以 后 执行 SQL 
指令 做 准备 。 

(2) 在 连接 到 MySQL 数据 库 服务 器 之 后 ， 接 下 来 使 用 mysql_select_db() 函 数 选择 数据 库 。 该 函 
数 的 语法 如 下 : 


mysql_select_db ( string 数据 库 名 [resource link_identifier] ) 


MySQL 服务 器 的 连接 标识 
传 入 MySQL 服务 器 的 数据 库 名 称 


mysql_query() 函 数 的 语法 如 下 : 


mysql_query("use 数据 库 名 "[,resource link_identifier]); 


如 果 没 有 指定 连接 标识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 的 连接 ， 本 函数 将 无 参数 调 
用 mysql_connectO 函 数 来 尝试 打开 一 个 数据 库 并 使 用 。 其 后 的 mysql_query0 函 数 调用 都 会 作用 于 活动 
数据 库 。 

例如 ， 在 本 模块 中 ,与 MySQL 数据 库 服务 器 中 的 db_jiari 数据 库 建 立 连接 。conn\conn.php 文件 的 
代码 如 下 : 


$conn=mysql_connect("127.0.0.1","root","111"); /连接 数据 库 服务 器 
mysql_select_db("db _jiari",$conn); /连接 数据 库 
mysql_query("set names utf8"); /| 设置 编码 
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2. MySQL 数据 操作 


对 数据 库 中 数据 的 操作 主要 是 使 用 mysql_query0 函 数 ， 本 模块 主要 使 用 mysql_query0) 函 数 实现 添 
加 和 删除 数据 的 操作 。 

(1) 添加 数据 

当 用 户 发 布 假日 公告 信息 时 ， 将 向 数据 库 中 添加 一 条 假日 信息 。promulgation_ok.php 文件 的 关键 
代码 如 下 : 


<?php 
header("content-type:text/html;charset=utf-8"); 
include("conn/conn.php"); 
ifisset($_POST[name]) and $_POST['Submit]==" 发 布 "X{ 
Sinsert=mysql_query("insert into tb_01(name,perform) values(".$ POST[name'].",".$_POST['radio].")", 


$conn); 1/ 添加 数据 
// 省 略 部 分 代码 
}else{ 
echo "<script>alert(' 请 选择 上 传 文件 ! ');window.location.href='promulgation.php';</script>"; 
?> 


(2) 删除 数据 
当 用 户 取 消 假日 公告 时 ， 将 删除 数据 库 中 的 假日 信息 。update.php 文件 的 代码 如 下 : 


<?php 
header("content-type:text/html;charset=utf-8"); 
include("conn/conn.php"); 
if(isset($_GET['abrogation])){ 
$update=mysql_query("delete from tb_01 where id=".$_GET[abrogation]."",$conn); 


if($updateX{ 
echo "<script>alert(' 假 日 信息 取消 成 功 ! ');window .location.href='cancel.php';</script>"; 
jelse{ 
echo "<script>alert(' 假 日 信息 取消 失败 ! ');window .location.href='cancel.php';</script>"; 
} 
} 
?> 


5.2.2 ”通过 超 链接 传递 值 
通过 超 链接 传递 值 主要 使 用 <a>…</a> 标 记 ， 其 语法 格式 如 下 : 
<a href=URL name=name target=target> 链 接 文字 </a> 


<a>…</a> 标 记 的 属性 如 下 。 

href 属性 : 指定 所 链接 文件 的 URL 路 径 。 该 路 径 可 以 是 相对 路 径 ， 也 可 以 是 绝对 路 径 。 

name 属性 : 指定 页 面 的 锚 点 名 称 ， 如 果 想 要 链接 到 对 应 的 锚 点 位 置 ， 需 要 在 锚 点 名 称 前 添加 
一 个 “#” 字 符 。 
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target 属性 : 指定 要 打开 的 链接 所 使 用 的 浏览 器 窗口 名 称 ， 可 以 使 用 自 定义 的 窗口 名 称 ， 也 可 
以 使 用 下 面 4 个 内 置 的 窗口 名 称 ， 其 中 前 两 个 比较 常用 。 
> _self: 在 当前 窗口 中 打开 链接 文件 ， 是 默认 值 。 
> _blank: 开启 一 个 新 的 窗口 打开 链接 文件 。 
> _parent: 在 父 级 窗口 中 打开 文件 ， 常 用 于 框架 页 面 。 
> _top: 在 顶层 窗口 中 打开 文件 ， 常 用 于 框架 页 面 。 
本 模块 主要 在 取消 公告 及 删除 公告 文件 中 使 用 了 超 链 接 。 
假日 公告 信息 取消 页 中 的 “取消 ” 超 链接 的 代码 如 下 : 


<a href="update.php?abrogation=<?php echo $array['id'];?>"> 取 消 </a> 


假日 公告 信息 管理 页 中 的 “删除 ” 超 链接 的 代码 如 下 : 
<a href='delete.php?catalog=".urlencode($catalog)."&filename=" . urlencode(getcwd ()) . "title=' 删 除 目录 或 者 文 
件 ' > 删除 </a> 
SC 人 
”获取 超 链接 参数 传递 的 值 ， 应 用 的 是 $_ GET[] 全 局 变量 。 


5.2.3 单 文 件 上 传 


本 模块 中 上 传 的 单 文件 是 公告 信息 文件 ， 主 要 使 用 了 mkdir0 和 move_uploaded_file() 函 数 ， 下 面 分 
别 对 其 进行 介绍 。 

(1) mkdir0 函 数 

该 函数 用 来 判断 某 文件 是 否 存在 ， 并 且 是 否 可 写 ， 如 果 满 足 上 述 条 件 则 返回 tue， 否 则 返回 false。 
该 函数 的 语法 如 下 : 


bool mkdir(string pathname [, int mode]) 
mkdir() 函 数 的 参数 说 明 如 表 5.1 所 示 。 
表 5.1 mkdir() 函 数 的 参数 说 明 


参数 说 了 明 
pathname 必要 参数 。 用 于 指定 新 建 目录 的 名 称 
mode 可 选 参数 。 用 于 指定 新 建 日 录 的 模式 


(2) move_uploaded_file() 函 数 
该 函数 应 用 POST 方法 实现 文件 的 上 传 。 其 语法 如 下 : 


bool move_uploaded file(string filename, string destination) 


move_uploaded_file() 函 数 的 参数 说 明 如 表 5.2 所 示 。 
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表 5.2 move_uploaded file() 函 数 的 参数 说 明 


参数 说 有明 
filename 必要 参数 。 指 定 要 上 传 的 文件 地 址 
destination 必要 参数 。 上 传 到 服务 器 后 的 存储 目录 及 名 称 


人 
"multipart/form-data"” 。 


5.2.4 多 图 片上 传 


本 模块 在 上 传 公告 信息 图 片 时 , 采用 多 图 片上 传 方式 ,使 用 for 循环 语句 循环 获取 图 片 信息 ,然后 
再 使 用 move_upload _file0) 函 数 执行 上 传 操 作 。 
for 语句 是 PHP 中 最 复杂 的 循环 控制 语句 ， 拥 有 3 个 条 件 表 达 式 。 其 语法 如 下 : 


for (expr1; expr2; expr3){ 
statement 


} 


for 循环 语句 的 参数 说 明 如 表 5.3 所 示 。 
表 5.3 for 循环 语句 的 参数 说 明 


必要 参数 。 第 1 个 条 件 表达 式 ， 在 第 1 次 循环 开始 时 被 执行 
必要 参数 。 第 2 个 条 件 表达 式 ， 在 每 次 循环 开始 时 被 执行 ， 决 定 循环 是 否 继续 
必要 参数 。 第 3 个 条 件 表达 式 ， 在 每 次 循环 结束 时 被 执行 

statement 必要 参数 。 满 足 条 件 后 ， 循 环 执行 的 语句 


其 执行 的 过 程 是 : 首先 执行 表达 式 1， 然 后 执行 表达 式 2， 并 对 表 
达 式 2 的 值 进行 判断 ， 如 果 值 为 真 ， 则 执行 for 循环 语句 中 指定 的 内 婴 
语句 〈 如 果 值 为 假 ， 则 直接 结束 循环 ， 跳 出 for 循环 语句 ) ， 最 后 执行 
表达 式 3〈 切 记 是 在 表达 式 2 的 值 为 真 时 ) ， 并 返回 表达 式 2 继续 循环 
执行 。for 循环 语句 的 操作 流程 如 图 5.8 所 示 。 


5.2.5 读 取 指 定 目录 中 的 目录 及 文件 


读 取 指 定 目录 中 的 目录 及 文件 主要 使 用 opendir0 、getcwd0 和 iconv0 
函数 ， 以 及 while 循环 语句 ， 循 环 输出 目录 文件 。 
(1) opendir0 函 数 
该 函数 用 来 打开 指定 的 目录 ， 并 返回 该 目录 的 标识 。 其 语法 如 下 : 图 5.8 ”for 循环 语句 的 流程 图 


resource opendir(string path) 


参数 path 为 该 函数 所 要 打开 目录 的 路 径 。 
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(2) getcewd() 函 数 
该 函数 返回 当前 的 工作 目录 。 其 语法 如 下 : 
string getcwd(void) 
该 函数 的 参数 为 空 。 
(3) iconv0 函 数 
该 函数 将 一 种 已 知 的 字符 编码 格式 A 转换 成 另外 一 种 编码 格式 B。 其 语法 如 下 : 


string iconv( string in_charset, string out_charset, string str ) 


参数 说 明 如 下 。 

in_charset: 一 种 已 知 将 被 转换 的 编码 格式 A。 

out_charset: 另 一 种 将 要 转换 成 的 编码 格式 B。 

str: 指定 被 转换 的 字符 串 。 该 函数 转换 成 功 返 回 字 符 串 ， 和 否则 返回 false。 

(4) while 循环 语句 

while 循环 语句 的 作用 是 反复 地 执行 某 一 项 操作 ， 是 循环 控 
制 语句 中 最 简单 、 最 常用 的 一 种 。while 循环 语句 对 表达 式 的 值 
进行 判断 ， 当 表达 式 为 非 0 值 时 ,执行 while 语句 中 的 内 机 语句 ; 
当 表 达 式 的 值 为 0 时 ， 则 不 执行 while 语句 中 的 内 嵌 语 句 。 该 语 
名 的 特点 是 : 先 判断 表达 式 ， 后 执行 语句 。while 循环 控制 语句 
的 操作 流程 如 图 5.9 所 示 。 


其 语法 如 下 ; 图 5.9 while 循环 控制 语句 的 操作 流程 
while (expn){ 
statement; 让 先 判断 条 件 ， 当 条 件 满足 时 执行 语句 块 ， 否 则 
不 向 下 执行 % 
} 


只 要 while 表达 式 expr 的 值 为 tue， 就 重复 执行 嵌 套 中 的 statement 语句 ， 如 果 while 表达 式 的 值 
一 开始 就 是 包 lse， 则 循环 语句 一 次 也 不 执行 。 


5.2.6 使 用 jQuery 控制 弹出 公告 信息 


假日 公告 管理 系统 中 ， 控 制 公告 信息 的 输出 应 用 的 是 jQuery 技术 ， 下 面 对 具 体 应 用 的 方法 进行 
介绍 。 
(1) $0， 用 于 替代 documentgetElementById0)。 语 法 如 下 : 


var someElement = $("#myld"); 
例如 ， 通 过 jQuery 获取 DOM 对 象 中 的 元 素 ， 其 代码 如 下 : 


$("div m"); 
$("div.admin"); 
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S$("div #mysqlid "); 
$("table a",content); 
第 一 行 代码 获取 所 有 标签 下 的 m 元 素 。 第 二 行 代码 获取 class 为 admin 的 元 素 。 第 三 行 代码 获取 标 
签 下 面 id 为 mysqlid 的 元 素 。 第 四 行 代码 获取 content 为 上 下 文 的 table 里 面 所 有 的 连接 元 素 。 
(2) slideToggle(speed,[callback])， 通 过 高 度 变化 来 切换 所 有 匹配 元 素 的 可 见 性 ， 并 在 切换 完成 后 
可 选 地 触发 一 个 回调 函数 。 
这 个 动画 效果 只 调整 元 素 的 高 度 ， 可 以 使 匹配 的 元 素 以 滑动 的 方式 隐藏 或 显示 。 
回 ”参数 speed: 3 种 预定 速度 之 一 的 字符 串 〈"slow"、"normal" 或 "fast") 或 表示 动画 时 长 的 毫秒 
数值 (如 1000) 。 
可 选 参数 callback: 在 动画 完成 时 执行 的 函数 。 
例如 ， 用 600 毫秒 缓慢 地 将 段落 滑 上 或 滑 下 。 


$("p").slideToggle("slow"); 
例如 ， 用 200 毫秒 快速 地 将 段落 滑 上 或 滑 下 ， 之 后 弹出 一 个 对 话 框 。 
$("p").slideToggle("fast",function(){ 
alert("Animation Done."); 
D; 
(3)ready(fn), 当 DOM 载 入 就 绪 可 以 查询 及 操纵 时 绑 定 一 个 要 执行 的 函数 。 该 方法 是 window.load 
事件 的 替代 方法 。 语 法 如 下 : 
ready(fn) 
参数 血 是 在 DOM 就 绪 时 执行 的 函数 ， 可 以 为 该 参数 任意 起 一 个 名 字 ， 并 因此 可 以 不 再 担心 命名 
冲突 而 放心 地 使 用 $ 别 名 。 


el 
SS 在 使 用 ready(fn) 时 ， 必 须 确保 在 <body> 元 素 的 onload 事件 中 没有 注册 函数 ， 否 则 不 会 能 
发 $(document).ready() 事 件 。 可 以 在 同一 个 页 面 中 无 限 次 地 使 用 $(document).ready() 事 件 。 其 中 ， 注 
册 的 函数 会 按照 代码 中 的 先后 顺序 依次 执行 。 


例如 ， 通 过 $(document).ready(fn) 代 蔡 <body> 标 签 的 onload 事件 ， 其 代码 如 下 : 
$(document).ready(function(X{ 

alert("jQuery"); 
六 


使 用 $(document).ready() 的 简写 , 同时 内 部 的 jQuery 代码 依然 使 用 $ 作 为 别名 ， 而 不 必 管 全 局 的 $ 是 
什么 。 其 jQuery 代码 如 下 : 


jQuery(function($) { 
…// 可 以 在 这 里 继续 使 用 $ 作 为 别名 


六 


第 5 章 假日 公告 管理 模块 (PHP+jQuery+MySQL 实现 ) 


上 述 $(documenb.readyO) 的 简写 蔡 换 的 是 如 下 代码: 

<body onload="alert(jQuery'):"> ee 
(4) css(name), 访问 第 一 个 匹配 元 素 的 样式 属性 。 返 回 值 为 string， 参 数 name 为 要 访问 的 属性 名 称 。 

例如 ， 取 得 第 一 个 段落 的 color 样式 属性 的 值 ， 其 代码 如 下 : 

$("p").css("color"); 


(5) css(name,value)， 在 所 有 匹配 的 元 素 中 ,设置 一 个 样式 属性 的 值 。 数 字 将 自动 转化 为 像素 值 。 
参数 name 为 属性 名 ， 参 数 value 是 属性 值 。 
例如 ， 初 始 状 态 下 隐藏 下 拉 列 表 ， 其 代码 如 下 


$("#cs").css("display","none"); /初始 状态 使 城市 下 拉 列 表 不 可 见 


在 假日 公告 管理 系统 中 ， 需 要 在 输出 公告 信息 的 网 页 中 载 入 jQuery 库 文 件 ， 编 写 JavaScript 脚本 ， 
控制 公告 信息 的 输出 。 关 键 代码 如 下 : 


<script src="js/jquery.js"></script> 
<script> 
function showLayerNotice(){ 
$("#Layer110").slideToggle(1000); 


} 
$(document).ready(function(){ 
$("#Layer110").css("left", (screen.width-520)/2); 
setTimeout("showLayerNotice()", 1300) ; 
); 
</script> 
同时 还 要 创建 <div> 标 签 ， 输 出 公告 信息 。 在 输出 公告 信息 之 前 ， 要 读 取 数 据 库 中 的 数据 ， 判 断 是 
否 输 出 公告 信息 。 如 果 指 定 字段 值 为 1， 则 包含 上 传 的 .html 文件 ， 即 输出 公告 信息 ， 否 则 不 输出 <div> 
标签 中 的 内 容 。 关 键 代码 如 下 : 


<?php 


include("conn/conn.php"); /连接 数据 库 

$result=mysql_query("select perform from tb_01",$conn);// 执 行 查询 语句 

$myrow=mysql_fetch_array($result); /获取 查询 结果 

if($myrow[perform]==1){ 1/ 判断 指定 字段 的 值 ， 如 果 等 于 1 则 输出 公告 信息 ， 否 则 不 输出 
?> 


<div id="Layer110" 

style="position: absolute; border:3px solid #CCCCCC; left: 50px; top: 50px; z-index: 1; font-size: Opx; 
display: none"> 
<?php 

include("upfile/index.html"); // 包 含 公告 文件 
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5.3 数据库 设计 


5.3.1 数据 库 分 析 


根据 系统 的 功能 结构 找 出 需要 保存 的 信息 《也 可 以 理解 为 现实 世界 中 的 实体 ) ， 并 将 其 转化 为 原 
始 数据 〈 属 性 类 型 ) 形式 。 这 种 描述 现实 世界 的 概念 模型 可 以 使 用 E-R 图 来 表示 ,也 就 是 实体 E-R 图 。 
最 后 ， 将 E-R 图 转换 为 关系 数据 库 。 这 里 重点 介绍 两 个 E-R 图 。 


1. 假日 公告 信息 实体 

假日 公告 信息 实体 包括 id、 公 告 名 称 和 是 否 显示 3 个 属性 ， 其 E-R 图 如 图 5.10 所 示 。 
2. 管理 员 信息 实体 

管理 员 信息 实体 包括 id、 用 户 名 和 密码 3 个 属性 ， 其 E-R 图 如 图 5.11 所 示 。 


图 5.10 假日 公告 信息 实体 E-R 图 图 5.11 管理 员 信息 实体 E-R 图 
5.3.2 ”创建 数据 库 和 数据 表 


系统 E-R 图 设计 完成 后 ， 根 据 E-R 图 来 创建 数据 库 及 数据 表 ， 先 来 看 一 下 所 使 用 的 数据 表情 况 
如 图 5.12 所 示 。 


表 操作 记录 数 电 类 型 整理 
tb_01 [El 六 加 Xx 1 MySAM utfe_general_ci 
tb_user [El 六 加 XX 1 MySAM utfe_general_ci 
2 个 表 总 计 2 MySAM utf8_general_ci 


5.12 假日 公告 管理 模块 数据 表 


各 个 表 的 结构 和 说 明 介绍 如 下 。 
加 ”tb_01 (假日 公告 信息 表 ) 
假日 公告 信息 表 用 于 存储 发 布 的 假日 信息 ， 其 结构 如 图 5.13 所 示 。 


@. 
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加 服务 器 : localhost ， 昌 数据 库 : db_jiari y 国 表 :tb_01 
_ 园 训 览 ， 阿 结构 ,有 SOL 万 搜索 了: 英 入 固 导 出 形 Import 做 操作 [而 清空 


宇航 类 型 整理 屋 性 Nul 扶 认 祯 外 
Fu int(4) 否 auto_increment 围 卢 
厂 name varchar(50) ute_unicode_ci 理 围 疙 
厂 perform varchar(1) utf8_general_ci 否 Pa 


图 5.13 假日 公告 信息 表 
tb_user (管理 员 信息 表 ) 
管理 员 信息 表 主要 用 于 存储 管理 员 信息 ， 其 结构 如 图 5.14 所 示 。 
四 服务 器 : localhost ， 轧 数据 库 : db_jiari * 国 表 :tb_user 
_ 町 浏览 _ 国 妾 构 ， 剖 SOL 万 搜 索 ， 庆 括 入 孙 导 出 _ 胶 Import 允 折 作出 洁 空 


字段 类 型 整理 尾 性 Null 默认 上 额外 
厂 过 int(4) 否 auto_increment 2 
name varchar(10) utfe_unicode_ci 否 4 
厂 pwd varchar(10) utre_unicode_ci 理 Fd 


图 5.14 管理 员 信 息 表 
5.4 用 户 登 录 


5.4.1 用 户 登录 概述 


用 户 登录 模块 是 假日 公告 管理 系统 的 门户 ， 只 有 登录 成 功 后 ， 才 可 以 进入 系统 主页 面 ， 并 进行 发 
布 、 取 消 及 删除 假日 信息 等 操作 。 用 户 登 录 模 块 运行 结果 如 图 5.15 所 示 。 


图 515 用 户 登录 模块 运行 结果 
5.4.2 ”用 户 登 录 功能 实现 过 程 


当 用 户 输入 登录 信息 后 ， 首 先 判断 用 户 名 和 密码 是 否 为 空 ， 然 后 再 判断 用 户 名 和 密码 是 否 正 确 ， 
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如 果 正 确 则 登录 成 功 ， 进 入 系统 主页 面 ， 否 则 将 给 出 提示 信息 。 
(1) 判断 用 户 名 和 密码 是 否 为 室 ， 应 用 的 是 checkitO 自 定义 脚本 函数 ， 存 储 于 user.php 文件 中 。 
关键 代码 如 下 : 


<script language="javascript"> 
function checkit()f // 自 定义 函数 
if(form1.name.value=="")f // 判 断 用 户 名 是 否 为 空 
alert(" 请 输入 用 户 名 ""); 
form1.name.select(); 
retum false; 
} 
if(form1.pwd.value=="™"){ /| 判断 密码 是 否 为 空 
alert(" 请 输入 密码 人"); 
form1.pwd.select(); 
retum false ; 


retum true; 


</script> 


(2) 判断 用 户 名 和 密码 是 否 正 确 ， 如 果 正 确 则 登录 成 功 ， 否 则 将 提示 用 户 名 和 密码 不 正确 ， 其 操 
作 在 user.php 文件 中 完成 。 代 码 如 下 : 


<?php 
include("conn/conn.php"); // 包 含 数据 库 连 接 文 件 
if(isset($_POST[name') and $_POST['pwd]!=null{ // 判 断 用 户 名 和 密码 是 否 为 空 
$select=mysql_query("select * from tb_user where name=".$_POST[name].”and pwd=".$_POST 
[pwd]."",$conn); /执行 查询 操作 
if($row=mysql_num_rows($select)==1X{ 
$_SESSION[name']=$_POST[name]; 
echo "<script>alert( 登 录 成 功 ! ');window.location.href='indexs.php';</script>;"; 
jelse{ 
echo "<script>alert(' 用 户 名 和 密码 不 正确 ! ');window.location.href='user.php';</script>;"; 
} 


} 
学 和 
ep // 省 略 部 分 代码 

<input type="image" name="imageField" onclick= "return checkit();" src="images/user_05.gif" /> 
ng // 省 略 部 分 代码 


<input type="image" name="imageField2" onclick"returmn false;" src="images/user_07.gif’ /> 


5.5 假日 公告 发 布 


5.5.1 假日 公告 发 布 概述 


假日 公告 发 布 模块 用 于 发 布 假日 公告 信息 。 进 入 假日 公告 发 布 页 面 ， 填 写 假日 名 称 ， 并 选择 假日 


@ 
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文件 及 图 片 ， 然 后 单 击 “ 发 布 ” 按 钮 ， 即 可 完成 公告 信息 发 布 。 假 日 公告 信息 发 布 页 面 运行 效果 如 
图 5.16 所 示 。 


RE 
Mocuments and Settnos\administ Dl 


roocumens sm Seine Woks 
BEN， Oorments an Satine ha 

roocuments ond Sctne eee 

[Br | 


人 是 个 杏 


a A 
图 5.16 假日 公告 信息 发 布 页 面 运行 效果 


5.5.2 ”假日 公告 信息 发 布 实 现 过 程 


(1) 创建 form 表单 ， 以 POST 方式 将 数据 提交 到 promulgation_ok.php 文件 中 进行 处 理 ， 并 且 设 
置 “enctype="multipart/form-data"” 属 性 ， 同 时 添加 假日 名 称 、 假 日 文件 、 假 日 图 片 和 是 否 执 行 等 表单 
元 素 。 关 键 代 码 如 下 : 


<form action="promulgation_ok.php" method="post" enctype="multipart/form-data" name="form1" id="form1"> 
<table id="_01" width="800" height="600" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td colspan="3"><a href="indexs.php"><img src="images/manage_01.gif" alt="" width="800" height= 
"201" border="0" /></a></td> 
</tr> 
<tr> 
<td><img src="images/manage_02.gif" width="140" height="251" alt=” /></td> 
<td width="523" height="251" align="center" valign="top" background="images/manage_03.gif"><table 
width="416" height="232"> 
<tr> 
<td width="89" height="30"> 假 日 名 称 : </td> 
<td width="315" align="|left"><input name="name" type="text" size="40" /></td> 
</tr> 
<tr> 
<td height="30"> 假 日 文件 : </td> 
<td align="left"><input name="files" type="file" size="31" /></td> 
</tr> 
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<tr> 
<td height="30"> 假 日 图 片 1: </td> 
<td align="left"><?php 
for($a=0;$a<4;$a++){ 
?> 
<input type="file" name="photo_<?php echo $a;?>" /> 
<?php 


<tr> 
<td height="30"> 是 否 执行 :</td> 
<td><label> 
<input name="radio" type="radio" value="1" checked="checked" /> 
是 
<input type="radio" name="radio" value="0" /> 
否 </label></td> 
</tr> 
<tr> 
<td height="30" colspan="2"><div align="center"> 
<input type="submit" name="Submit" value=" 发 布 " /> 
<input type="reset" name="Submit2" value=" 取 消 " /> 
</div></td> 
</tr> 
</table></td> 
<td><img src="images/manage_04.gif" width="137" height="251" alt=" /></td> 
</tr> 
<tr> 
<td colspan="3"><img src="images/manage_05.gif" width="800" height="148" alt="" /></td> 
</tr> 
</table> 
</form> 


(2) 创建 promulgation_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 完 成 假日 公告 信息 的 发 布 操作 。 关 
键 代码 如 下 : 


<?php 
header("content-type:text/html;charset=utf-8"); 
include("conn/conn.php"); 
if(isset($_POST[name']) and $_POST['Submit]==" 发 布 "X{ 
Sinsert=mysql_query("insert into tb_01(name,perform) values(".$_ POST[name'].",".$_POST['radio].")", 


$conn); // 添 加 数据 
if(isset($_FILES[Yfiles]rname])X /独断 文件 及 图 片 是 否 存在 
if(lis_dir("./upfile/images")}{ // 判 断 指定 目录 是 否 存在 
mkdir("./upfile/images"); // 创 建 目录 
; 
$photo="upfile/".$_FILES['files"['name']; /文件 的 存储 路 径 和 名 称 


if(move_uploaded file($_FILES[files'][tmp_name],$photo)i{  // 执 行 上 传 
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for($a=0;$a<4;$a++){ // 循 环 语句 
$name=$_FILES[photo '.$a]; /将 变量 的 名 称 保存 在 变量 中 
$path='upfile/images/.$name[name']; /定义 上 传 文件 的 路 径 


move_uploaded file($nameftmp_name'],$path); 


echo "<script>alert( 上 传 成 功 ! ');window.location.href='promulgation.php';</script>"; 


}else{ 
echo "<script>alert( 上 传 失败 ! ');window.location.href='promulgation.php';</script>"; 
} 
S 
}else{ 
echo "<script>alert(' 请 选择 上 传 文件 ! ');window .location.href='promulgation.php';</script>"; 
} 
35 


5.6 假日 公告 取消 


5.6.1 假日 公告 取消 概述 


假期 过 后 ， 在 假日 取消 页 面 中 单 击 假日 名 称 后 的 “取消 ” 超 链接 ， 即 可 取消 该 假日 公告 ， 并 在 数 
据 库 中 删除 该 假日 信息 。 假 日 公告 信息 取消 页 面 运行 效果 如 图 5.17 所 示 。 


5.6.2 浏览 公告 信息 


打开 假日 信息 取消 页 面 ， 默 认 将 显示 所 有 假日 信息 。 通 过 select 查询 语句 ， 查 询 假日 公告 信息 表 
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中 的 数据 ， 通 过 while 语句 循环 输出 假日 公告 信息 ， 并 且 为 每 条 信息 设置 一 个 “取消 ” 超 链 接 ， 将 信息 
的 id 作为 参数 值 传递 到 update.php 文件 中 。cancel.php 的 关键 代码 如 下 : 


<?php 
include("conn/conn.php"); 
$select=mysql_query("select * from tb_01",$conn); 
while($array=mysql_fetch_array($select)){ 
?> 
<tr> 
<td height="30"><?php echo $array[name];?> </td> 
<td><div align="center"> 
<a href="update.php?abrogation=<?php echo $array['id'];?>"> 取 消 </a> 
</div></td> 
/ltr> 
<?php 
} 
?> 


5.6.3 ”取消 假日 公告 


取消 假日 公告 主要 是 通过 超 链接 参数 abrogation 传递 的 值 , 在 update.php 页 中 执行 delete 删除 语句 
删除 信息 ， 取 消 假日 公告 。update.php 的 关 健 代码 如 下 : 


<?php 
header("content-type:text/html;charset=utf-8"); 
include("conn/conn.php"); 
if(isset($_GET['abrogation])){ 
$update=mysql_query("delete from tb_01 where id=".$_GET[abrogation]."",$conn); 


if($updateX{ 
echo "<script>alert(' 假 日 信息 取消 成 功 ! ');window.location.href='cancel.php';</script>"; 
Jelse{ 
echo "<script>alert(' 假 日 信息 取消 失败 ! ');window.location.href='cancel.php';</script>"; 
} 
} 
?> 


5.7 假日 公告 信息 管理 


5.7.1 假日 公告 信息 管理 概述 


假日 公告 信息 管理 模块 对 上 传 的 公告 文件 及 公告 图 片 进行 管理 ， 将 过 期 的 公告 信息 文件 删除 ， 以 
免 造成 空间 浪费 。 假 日 公告 信息 管理 模块 运行 效果 如 图 5.18 所 示 。 
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5.7.2 ”假日 公告 信息 管理 实现 过 程 


(1) 创建 manage.php 动态 页 ， 输 出 指定 目录 下 的 文件 夹 及 文件 ， 并 实现 “上 级 目录 ”的 跳 转 。 
关键 代码 如 下 : 
<?php 


if (lisset ( $_GET [catalog] ) || empty ( $_GET [catalog] ) 
$current_directory = getcwd (); /获得 脚本 目录 


else 
$current_directory = $_GET ['catalog]; 
chdir ( iconv ( "utf-8", "gb2312", $current_directory ) ); /改变 当前 目录 
echo "<span class='STYLE4'> 当 前 目录 :</span><span class='STYLE2'>" .iconv ( "gb2312", "utf-8", getcwd () ) . 
"</span><br>"; 
$ml = opendir ( iconv ( "utf-8", "gb2312", $current_directory ) ); // 打 开 目 录 
while ( $gain_directory = readdir ( $ml ) ) { // 循 环 读 取 目 录 中 的 文件 夹 及 文件 
echo "<tr><td align='center valign="'middle’>"; 
if (is_dir ( $gain_directory )) { // 判 断 是 目录 
if ($gain_directory == ".") { 
$catalog = getcwd (); /显示 当前 目录 
$catalog = iconv ( "gb2312", "utf-8", $catalog ); 
echo "<a href=manage.php?catalog=".urlencode($catalog)."><span class='style1'> 锁 定 </span> 


</a>"; 
} elseif ($gain_directory == "..") { 
$catalog = getcwd () . \.."; // 上 级 目录 
$catalog = iconv ( "gb2312", "utf-8", $catalog ); 
if($catalog==""X 
echo "<a href=manage.php?catalog=".urlencode($catalog)."><span class='style1> 上 级 目 
录 </span></a>"; 


Jelse{ 
echo "<a href=manage.php?catalog=".urlencode($catalog)."><span class='style1> 上 级 目 
录 </span></a>"; 
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} 
}else{ 
$catalog = getcwd () . \$gain_directory"; // 子 目录 
$catalog = iconv ( "gb2312", "utf-8", $catalog ); 
echo "<a href=manage.php?catalog=".urlencode($catalog).">" . iconv ( "gb2312", "utf-8", $gain_ 
directory ) . "</a>"; 


: 
}else{ 
$ext = substr ( $gain_directory, strrpos ( $gain_directory, "." ) ); 
if (strtoupper ( $ext ) == ".html" || strtoupper ( $ext ) == ".gif" || strtoupper ( $ext ) == "jpg" ) { 


S$catalog = getcwd (); 
$catalog = iconv ( "gb2312", "utf-8", $catalog ); 
echo iconv ("gb2312", "utf-8", $gain_directory ) ; 
}else{ 
echo iconv ( "gb2312", "utf-8", $gain_directory ); 
二 
if (is_dir ( $gain_directory )) 
$file_size = "目录 "; 
else 
Sfile_size = filesize ( $gain_directory ); 
echo "<td align='center valign='middle'>$file_size</td>"; 
$create time = date ( "Y-m-d H:i:s", filectime ( $gain_directory ) ); 
echo "<td align='center valign='middle’>" . iconv ( "gb2312", "utf-8", $create_time ) . "</td>"; 
S$update_time = date ( "Y-m-d H:i:s", filemtime ( $gain_directory ) ); 
echo "<td align='center valign='middle’>$update_time</td>"; 
echo "<td align='center valign='middle’>"; 
if ($gain_directory == ".") { 
$catalog = getcwd (); /显示 当前 目录 
echo "删除 "; 
} elseif ($gain_directory == "..") { 
$catalog = getcwd () .\."; /上 级 目录 
echo "删除 "; 
}else{ 
$catalog = getcwd () . "\\$gain_directory"; // 子 目录 
echo "<a href='delete.php?catalog=".urlencode($catalog)."&filename=" . urlencode(getcwd ()) . ” 
title=' 删 除 目录 或 者 文件 ' > 删除 </a>"; 
} 
echo "</td>"; 
echo "</tr>"; 


} 
closedir ( $ml ); 
?> 


(2) 创建 delete.php 页 ， 当 用 户 单 击 “ 删 除 ” 超 链接 时 , 在 delete.php 页 面 接收 超 链接 传递 的 参数 ， 
完成 删除 操作 。 关 键 代码 如 下 : 


<?php 
header("content-type:text/html;charset=utf-8"); 


&@. 
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$count=substr_count($_GET['catalog"],".); 
if($count>=1){ 
if(unlink($_GET['catalog'"])){ 
echo "<script>alert(' 文 件 删除 成 功 !"); window.location.href='manage.php?catalog=".urlencode($_GET 
['filename']).";</script>"; 


jelse{ 
echo "<script>alert( 文 件 删除 失败 ! '); history.back();</script>"; 
$ 
}else{ 


iflis_dir($_GET['catalog'])){ 
if(@rmdir($_GET['catalog])\{ 
echo "<script>alert(' 目 录 删 除 成 功 ! '); window.location.href='manage.php?catalog=".urlencode 
($_GET['filename').";</script>"; 
}else{ 
echo "<script>alert( 目录 删除 失败 ! '); history.back();</script>"; 
上 


} 


?> 


/ 
所 四 在 本 模块 中 ,创建 indexs php 文件 ， 模 拟 网 站 主页 ， 当 完成 假日 公告 信息 的 添加 之 后 ， 运 
行 indexs.php 文件 ， 将 弹出 公告 信息 。 其 运行 效果 如 图 5.19 所 示 。 


尊敬 的 读者 朋友 们 : 
由 于 新 春 佳节 假日 临近 ， 本 公司 于 2011 年 1 月 31 日 到 

2011 年 2 月 6 日 放假 , 所 以 在 此 期 间 暂 停 服务 , 请 将 您 的 问 

题 发 到 技术 论坛 或 邮箱 中 ， 上 班 后 我 们 会 第 一 时 间 回复 , 

给 您 带 来 的 不 便 ， 请 谅解 ! 


5.19 弹出 公告 信息 效果 


5.8 本 章 小 结 


本 章 通过 一 个 完整 的 假日 公告 管理 系统 带领 读者 详细 讲解 一 个 模块 的 开发 流程 。 同 时 ， 在 假日 公 
告 管理 系统 中 ， 采 用 jQuery 控制 弹出 公告 信息 ， 以 及 单 文件 上 传 和 多 图 片上 传 的 功能 ， 使 页 面 显示 得 
更 加 有 条 理性 。 通 过 本 章 的 学 习 ， 读 者 不 仅 可 以 了 解 一 些 网 站 的 基本 流程 ， 而 且 可 以 熟悉 添加 和 删除 


公告 信息 的 实现 方法 。 
@ 
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分 页 显示 模块 


(PHP+Ajax+Smarty+PDO 实现 ) 


在 网 站 中 ， 当 面 对 数 据 库 中 的 大 量 数据 要 输出 时 ， 在 头脑 中 闪 过 
的 第 一 个 想法 就 是 应 用 什么 分 页 技术 输出 数据 库 中 的 数据 才 更 加 合 
理 、 简 单 、 快 捷 。 分 页 功能 是 每 个 网 站 都 必 不 可 少 的 ， 在 本 模块 中 ， 
将 介绍 几 种 分 页 的 实现 方法 ， 项 望 对 读者 以 后 程序 的 开发 有 所 帮助 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

WI 自 定 义 轩 数 蔡 换 超 长 文本 中 的 特殊 字符 

WI 自 定 义 轩 数控 制 超 长 文本 的 输出 

mm ”PDO 操作 MySQL 数据 库 

MH ”PHP 起 长 文本 分 页 

MW ”Ajax 无 刷新 分 页 

让“ 分 页 类 分 页 

# PHP 上 下 分 页 

MH ”PHP 跳 转 分 页 
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6.1 分 页 显示 概述 


6.1.1 模块 概述 


分 页 显示 模块 是 网 站 开发 中 最 常用 的 一 个 典型 模块 ， 该 模块 的 最 大 用 途 就 是 将 海量 数据 或 者 超 长 文 
本 数据 分 页 输出 。 这 样 一 来 ， 不 仅 使 页 面 的 整体 效果 更 加 紧凑 ， 而 且 能 带 来 最 好 的 用 户 体验 。 


6.1.2 ”功能 结构 


在 本 模块 中 ,整体 上 分 为 两 部 分 内 容 : 一 部 分 是 面向 对 人 机 
象 的 分 页 ; 另 一 部 分 是 面向 过 程 的 分 页 。 在 面向 对 象 分 页 中 ， | 
通过 Smarty 模板 完成 网 页 的 动静 分 离 ， 通 过 PDO 操作 | 
MySQL 数据 库 , 并 且 将 Smarty 模板 的 配置 、 PDO 连接 操作 图 。 本 
数据 库 以 及 分 页 方法 都 封装 到 类 中 , 通过 类 中 方法 的 调用 完 x| || sl 
成 各 种 操作 ， 在 面向 过 程 的 分 页 中 ， 完 成 超 长 文本 的 分 页 、 | | 人 | ml 区 
Ajax 无 刷新 分 页 、 跳 转 分 页 和 上 下 分 页 。 本 模块 的 系统 功能 分 区 
结构 如 图 6.1 所 示 。 | 

吕 1 
6.1.3 程序 预览 图 6.1 分 页 显示 模块 功能 结构 图 


在 分 页 显示 模块 中 ， 展 示 了 5 种 不 同 的 分 页 方法 : 超 长 
文本 的 分 页 显示 、Ajax 无 刷新 分 页 、 跳 转 分 页 、 上 下 分 页 和 通过 分 页 类 分 页 。 
超 长 文本 的 分 页 显示 : 实现 对 文本 文件 中 数据 的 分 页 显示 ， 其 主要 应 用 在 注册 须知 、 公 司 简 
介 、 版 权 说 明 等 文本 类 的 文件 输出 中 ， 如 图 6.2 所 示 。 
Ajax 无 刷新 分 页 : 实现 数据 库 中 数据 的 无 刷新 显示 ， 其 主要 应 用 在 一 些 支持 视频 播放 的 网 
中 ， 通 过 Ajax 无 刷新 进行 分 页 ， 不 会 影响 到 视频 文件 的 播放 ， 如 图 6.3 所 示 。 


[HE 国 半 为 名 而 “人 如 入 了 于 和 


等 


[三 县 英国 肌 及 的 多 全 a 人 


四 琶 交 区 向 VG thns | 间 ， CT 局 
La | wi ? 损 可 加 几 朋 DIE 荐 四 代 本 FE] 骨 中 亲人 
编程 癌 内。 有 用 扣 。。。。 鞭 央 费 折 版 权 志 明日 简介 SSE 


OT ER [3 


图 6.3 ”Ajax 无 刷新 分 页 的 应 用 
跳 转 分 页 : 实现 在 当前 页 面 中 到 其 他 指定 页 面 的 跳 转 ， 通 过 它 与 标准 分 页 功能 的 结合 使 用 可 


_- 国 
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以 增加 分 页 操作 的 灵活 性 。 如 图 6.4 所 示 。 
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图 64 PHP 跳 转 分 页 的 运行 结果 
上 下 分 页 : 针对 当前 的 页 面 ， 实 现 上 一 页 和 后 一 页 的 跳 转 ， 也 属于 一 种 常用 的 分 页 方法 ， 如 
图 6.5 所 示 。 
通过 分 页 类 分 页 : 将 一 种 定义 好 的 分 页 方法 封装 到 一 个 类 中 ， 当 需要 进行 分 页 操作 时 ， 可 以 
直接 调用 这 个 类 文件 ， 并 设置 相应 的 参数 ， 即 可 实现 分 页 的 操作 ， 无 须 再 重新 编写 分 页 方法 。 
这 是 一 个 比较 实用 的 分 页 方法 ， 如 图 6.6 所 示 。 
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图 6.5 PHP 上 下 分 页 的 运行 结果 图 6.6 分 页 类 分 页 的 运行 结果 


6.2 面向 对 象 分 页 


通过 分 页 类 实现 分 页 功能 是 一 个 非常 好 的 方法 。 一 个 分 页 类 创建 成 功 后 ， 就 可 以 直接 调用 类 文件 
实现 分 页 功能 , 而 无 须 再 重新 编译 。 在 分 页 类 模块 中 , 采用 Smarty 模板 完成 网 页 的 动静 分 离 , 通过 PDO 
完成 对 数据 库 的 连接 和 操作 。 分 页 类 实现 分 页 的 运行 结果 如 图 6.6 所 示 。 


6.2.1 Smarty 模板 的 安装 和 配置 
将 Smarty 模板 的 压缩 包 解压 ， 复 制 libs 目录 到 服务 器 MR\06\01\system 目录 下 。 


@ 
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配置 Smarty 模板 ， 在 确定 Smarty 模板 文件 的 位 置 之 后 ， 在 smarty 目录 下 新 建 templates、templates_c、 
configs 和 cache 4 个 目录 ， 用 于 存储 不 同 的 文件 。 

封装 Smarty 模板 的 配置 类 ,完成 Smarty 模板 的 配置 操作 ， 该 类 存储 于 06\01\system\system.smarty. 
inc.php 文件 中 。 其 代码 如 下 : 


<?php 

require("libs/Smarty.class.php"); // 包 含 模板 文件 

class SmartyProject extends Smarty{ /| 定义 类 ， 继 承 模板 类 
function SmartyProject(){ /定义 方法 
S$this->template_dir = "./"; /指定 模板 文件 存储 位 置 
$this->compile_dir = "/system/templates_c/"; /指定 编译 文件 存储 位 置 
$this->config_dir = "./system/configs/"; /指定 配置 文件 存储 位 置 
$this->cache_dir = "./system/cache/"; /指定 缓存 文件 存储 位 置 
} 

} 

?> 


6.2.2 加载 PDO 数据 库 抽象 层 


在 本 模块 中 , 通过 PDO 数据 库 抽 象 层 连接 和 操作 MySQL 数据 库 。 PDO 是 与 PHP 5.1 一 起 发 行 的 ， 
默认 包含 在 PHP 5.1 中 。 由 于 PDO 需要 PHP 5 核心 面向 对 象 特性 的 支持 ， 因 此 其 无 法 在 PHP 5.0 之 前 
的 版 本 中 使 用 。 

默认 情况 下 ，PDO 在 PHP 5.2 中 为 开启 状态 ， 但 是 要 启用 对 某 个 数据 库 驱 动 程序 的 支持 ， 仍 需要 
进行 相应 的 配置 操作 。 

在 Linux 环境 下 ， 要 使 用 MySQL 数据 库 ， 可 以 在 configure 命令 中 添加 如 下 选项 : 

--With-pdo-mysql=/path/to/mysql/installation 


在 Windows 环境 下 ，PDO 在 Bhp ini 文 件 中 进行 配置 ， 如 图 6.7 所 示 。 


extension=php_ 后 由 
extension=php_pdo_mysql. 
;extension=php_pdo_ Pe dl 
extension=php_pdo_mssql. 


eztension=php_pdo_oci. dll 
extension=php_pdo_oci8. dll 
ertension=php_pdo_odbe. dl1 
;extension=php_pdo_pesql. dl 
ertension=php_pdo_sqlite. dll 


6.7 ”Windows 环境 下 配置 PDO 


要 启用 PDO， 首 先 必须 加 载 “extension=php_pdo.dl1”， 如 果 要 想 其 支持 某 个 具体 的 数据 库 ， 那 么 
还 要 加 载 对 应 的 数据 库 选项 。 例 如 , 要 支持 MySQL 数据 库 , 则 需要 加 载 “extension=php_pdo_mysql.dll” 
选项 。 


Ma a 
生效 。 有关 PDO 连接 和 操作 MySQL 数据 库 的 方法 将 在 技术 提炼 中 做 详细 讲解 ， 这 里 不 再 装 述 。 
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6.2.3 ”分 页 类 模块 的 页 面 设计 


由 于 本 模块 的 开发 使 用 的 是 Smarty 模板 技术 ， 因 此 模块 的 页 面 设 计 和 程序 的 开发 是 完全 分 离 的 ， 
模块 的 页 面 设计 是 在 index.tpl 文件 中 完成 的 ， 而 程序 的 开发 则 是 在 index.php 文件 中 完成 。 

分 页 类 模块 的 页 面 设计 主要 就 是 将 程序 开发 中 模板 变量 中 的 数据 进行 输出 ， 以 及 分 页 功能 的 实 
现 。 关 键 代码 如 下 : 


{if $isbbs=="T"}<!-- 首 先 通过 模板 变量 $isbbs 判断 是 否 有 数据 --> 

<!-- 通 过 section 内 建 函 数 ， 循 环 输出 数组 模板 变量 garraybbs 中 的 图 书 数据 --> 

{php} $i=1 {/php} 

{section name=bbsid loop=$arraybbs} 

<tr> 

<td height="22" class="STYLE4" {php} if($i%2==0){ echo "bgcolor='#F5F5F5";H/php}>&nbsp; 
<a href="index.php?id={$arraybbs[bbsid].tb_book_id}" target="_blank"> 

{unhtml content=$arraybbs[bbsid]. tb_book_name}</a> 

</td> 

<td class="STYLE4" {php} if($i%2==0){ echo "bgcolor=#F5F5F5";/php}> 

<div align="center">{$arraybbs [bbsid].tb_book_synopsis}</div></td> 

<td height="22" class="STYLE4" {php} if($i%2==0){ echo"bgcolor=#F5F5F5";M/php}> 
<div align="center"> {$arraybbs[bbsid].tb_print_date}</div></td> 

<td height="22" class="STYLE4" {php} if($i%2==0){ echo "bgcolor=#F5F5F5";}/php}> 
<div align="center"> {$arraybbs[bbsid].tb_book_author}</div></td> 

</tr> 

{php} $i++; Uphp} 

{/section} 

<table width="100%" height="22" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr><!-- 输 出 分 页 功能 的 信息 --> 

<td class="STYLE4">&nbsp;{$showpage}</td> 

</tr> 

</table> 


Wi 

<!- 判 断 当 没有 数据 时 ， 则 输出 如 下 内 容 --> 

{if $isbbstell=="F" && $isbbs=="F"} 

<table width="96%" height="30" border="0" align="center" cellpadding="0" cellspacing="1" bgcolor="#58BAE9"> 
<tr> 

<td bgcolor="#FFFFFF"><div align="center"> 

<font color="#FF0000">{Idelim} 该 分 类 暂 无 论坛 公告 或 帖子 {rdelim}</font></div></td> 

</tr> 

</table> 


{i 
6.2.4 “分 页 类 模块 的 程序 开发 
分 页 类 模块 的 程序 开发 是 在 index.php 文件 中 进行 的 ， 是 一 个 纯粹 的 PHP 代码 页 。 在 该 文件 中 调 


用 类 文件 ， 实 现 与 数据 库 的 连接 ， 读 取 数据 库 中 的 数据 ;调用 分 页 类 ， 实 现 数据 的 分 页 显示 ， 并 且 定 
义 模板 变量 ， 用 于 在 模板 页 中 输出 数据 。 关 键 代码 如 下 : 


@ 
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<?php 
header("Content-Type:text/html;charset=gb2312"); 
require_once("system/system.inc.php");”// 调 用 指定 的 文件 


S$arraybbstell=$admindb->ExecSQL("select * from tb_mr_book",$conn); /| 执行 select 查询 语句 
if(!$arraybbstell){ 

$smarty->assign("isbbstell","F"); // 判 断 如 果 执 行 失败 ， 则 输出 模板 变量 isbbstell 的 值 为 F 
}else{ 

$smarty->assign("isbbstell","T"); /| 判断 如 果 执 行 成 功 ， 则 输出 模板 变量 isbbstell 的 值 为 下 


$smarty->assign("arraybbstell", $arraybbstell); // 定 义 模板 变量 arraybbstell， 输 出 数据 库 中 的 数据 


} 
iflisset($_GET["page"])){ 
$page=$_GET["page"]; 
}else{ 

$page=1; 


J 

$arraybbs=$seppage->ShowData("select * from tb_mr_book",$conn,1,$page); // 调 用 分 页 类 ， 实 现 分 页 功能 
if(l$arraybbs){ 

$smarty->assign("isbbs","F"); 

}else{ 

$smarty->assign("isbbs","T"); 

$smarty->assign("showpage",$seppage->ShowPage(,",",”",")); /定义 输出 分 页 数据 的 模板 变量 ShowPage 
$smarty->assign("arraybbs",$arraybbs); 


$smarty->display("index.tpl"); /指定 连接 的 模板 页 
Ty 


至 此 ， 分 页 类 模块 的 讲解 完毕 ， 程 序 的 完整 代码 请 参考 本 章光 得 中 的 内 容 。 
6.3” 超 长 文本 分 页 


6.3.1 超 长 文本 分 页 功能 概述 


通过 PHP 超 长 文本 的 分 页 显示 功能 可 以 对 指定 文本 文件 中 的 数据 进行 分 页 输出 .运行 结果 如 图 6.8 
所 示 。 
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图 6.8 PHP 超 长 文本 分 页 显示 的 运行 结果 


PHP 典型 模块 开发 全 程 实录 


6.3.2 ” 超 长 文本 分 页 功能 实现 过 程 


PHP 超 长 文本 的 分 页 显示 主要 应 用 的 是 自 定义 函数 unhtml0 和 msubstr0， 有 关 这 两 个 自 定 义 函 数 
的 创建 请 参考 6.8.1 节 和 6.8.2 节 中 的 内 容 ， 这 里 不 再 獒 述 。 

PHP 超 长 文本 分 页 显示 的 操作 在 06\mr_copyright.php 文件 中 完成 。 首 先 ,判断 变量 的 值 是 否 为 空 ， 
并 调用 指定 的 包含 文件 ， 获 取 自 定义 函数 。 代 码 如 下 : 

if(isset($_POST[page]))f 

$page=$_POST['page'; 

}else if(isset($_GET[page'"])X{ 

$page=$_GET[page]; 

}else{ 

$page=1; 


} 
include("function.php"); // 包 含 自 定义 函数 


然后 ， 通 过 file_get_contents() 函 数 获取 file.txt 文本 文件 中 的 数据 ， 返 回 到 一 个 字符 串 中 ; 通过 
unhtml0) 函 数 蔡 换 字 符 串 中 的 特殊 字符 ; 通过 strlen() 函 数 获取 字符 串 的 长 度 ， 通 过 ceil() 函 数 计算 将 文 
本 文件 中 的 数据 分 成 儿 页 进行 输出 ; 通过 msubstr0 函 数控 制 数据 的 输出 ; 应 用 substr() 函 数 获取 到 当前 
页 输出 的 内 容 。 代 码 如 下 : 


<?php 

if($pageX{ 

$counter=file_get_contents("file/file.txt"); // 读 取 文 本 文件 中 的 数据 ， 返 回 一 个 字符 串 
S$length=strien(unhtml($counter)); // 计 算 字符 串 的 长 度 ， 替 换 特殊 字符 
$page_count=ceil($length/1050); // 对 字符 串 进行 分 页 处 理 
$c=msubstr($counter,0,($page-1)*1050); // 通 过 自 定义 函数 获取 上 一 页 中 的 数据 
$c1=msubstr($counter,0,$page*1050); // 通 过 自 定义 函数 获取 当前 页 中 的 数据 
echo substr($c1,strlen($c),strlen($c1)-sten($c)); /获取 到 当前 页 中 要 输出 的 数据 

} 

?> 


最 后 ， 创 建 分 页 超 链接 ， 实 现 首 页 、 上 一 页 、 下 一 页 和 尾 页 之 间 的 跳 转 。 代 码 如 下 : 


<td width="42%" align="center" valign="middle" bgcolor="#FFFFFF"> 

<span class="STYLE1">&nbsp;&nbsp; 页 次 : <?php echo $page;?> / <?php echo $page_count;?> 页 </span></td> 
<td width="58%" height="28" align="left" valign="middle" bgcolor="#FFFFFF"> 

<span class="STYLE1"> &nbsp; 分 页 : 

<?php 

if($page!=1X{ 

echo "<a href=index.php?Imbs= 了 明日 版 权 声 明 &page=1> 首 页 </a>&nbsp;"; 

echo "<a href=index.php?Imbs= 阴 日 版 权 声明 &page=".($page-1)."> 上 一 页 </a>&nbsp;"; 
} 

if($page<$page_count}{ 

echo "<a href=index.php?Imbs= 阴 日 版 权 声明 &page=".($page+1)."> 下 一 页 </a>&nbsp;"; 
echo "<a href=index.php?Imbs= 阴 日 版 权 声 明 &page=".$page_count."> 尾 页 </a>"; 


他 
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下 
?> 
</span> </td> 


至 此 ，PHP 超 长 文本 的 分 页 显示 介绍 完毕 。 
6.4 Ajax 无 刷新 分 页 


6.4.1 Ajax 无 刷新 分 页 功能 概述 


Ajax 无 刷新 分 页 的 强大 功能 体现 最 为 明显 的 地 方 就 是 在 播客 中 。 因 为 在 播客 中 ， 在 播放 视频 文件 
的 同时 ， 如 果 使 用 的 不 是 无 刷新 分 页 ， 那 么 当 执 行 翻 页 的 操作 后 ， 视 频 文件 将 被 重新 打开 ;如果 使 用 
的 是 无 刷新 分 页 ， 就 不 会 出 现 这 种 情况 。 

为 了 更 好 地 理解 Ajax 无 刷新 分 页 的 强大 功能 ， 这 里 创建 了 一 个 视频 文件 ， 并 且 应 用 Ajax 无 刷新 
分 页 输出 文本 文件 中 的 内 容 ， 验 证 无 刷新 分 页 的 操作 是 否 对 视频 文件 的 播放 有 影响 。 运行 结 果 如 图 6.9 
所 示 。 
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图 6.9 Ajax 无 刷新 分 页 的 应 用 
6.4.2 Ajax 无 刷新 分 页 功能 实现 过 程 


Ajax 无 刷新 分 页 由 mr_synopsis.php、mr_synopsis_OK.php 和 discuss_js.js 3 个 文件 组 成 。 

在 mr_synopsis.php 文件 中 ， 首 先 ， 创 建 一 个 div 标签 ， 设 置 div 标签 的 id="synopsis"。 然 后 ， 在 
div 标签 中 编写 超 长 文本 分 页 输出 的 代码 ， 并 且 在 超 长 文本 分 页 的 超 链 接 中 应 用 onclick 事件 ， 调 用 自 定 
义 函 数 artpagination()， 传 递 的 参数 为 另外 一 个 实现 分 页 功能 的 文件 mr_synopsis_ok.php。 关 键 代码 如 下 : 


<div id="synopsis"><!-- 创 建 div 标签 ， 用 于 获取 js 文件 中 返回 的 分 页 结果 --> 
<table width="425" border="0" cellspacing="0" cellpadding="0"> 


_ 国 
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<tr> 

<td height="20">&nbsp;</td> 

</tr> 

<tr> 

<td> 

<?php 

// 读 取 超 长 文本 中 的 数据 ， 实 现 超 长 文本 中 数据 的 分 页 显示 

if($pageX{ 

$counter=file_get_contents("file/mr_synopsis.txt"); 

$length=strlen($counten); 

$page_count=ceil($length/950); 

$c=msubstr($counter,0,($page-1)*950); 

$c1=msubstr($counter,0,$page*950); 

echo substr($c1,strlen($c),strlen($c1)-stren($c)); 

</td> 

</tr> 

<tr> 

<td> 

<table border="0" align="center" cellpadding="0" cellspacing="0"> 

<tr><!-- 设 置 超 长 文本 分 页 显示 的 超 链接 -> 

<td width="42%" align="center" valign="middle" bgcolor= "十 FFFFF"> 

<span dlass="STYLE1">&nbsp;&nbsp; 页 次 : <?php echo $page;?> / <?php echo $page_count;?> 页 </span></td> 
<td width="58%" height="28" align="left" valign="middle" bgcolor="#FFFFFF"> 

<span class="STYLE1"> &nbsp; 分 页 : 

<?php 

if($page!=1X{ 

?> 

<!-- 调 用 artpagination() 函 数 ， 实 现 无 刷新 的 分 页 输出 -> 

<a href="#" onclick='return artpagination("mr_synopsis_ok.php?page=1")'> 首 页 </a>&nbsp; 
<a href="#" onclick='return artpagination("mr_synopsis_ok.php?&page=<?php echo $page-1;?>")> 上 一 页 </a> 
&nbsp; 

<?php } 

if($page<$page_countjf 

?> 

<a href="#" onclick='return artpagination("mr_synopsis_ok.php?page=<?php echo $page+1;?>")> 下 一 页 </a> 
&nbsp; 

<a href="#" onclick='return artpagination("mr_synopsis_ok.php?page=<?php echo $page_count'?>")> 尾 页 </a> 
<?php 

?> 

</span> </td> 

</tr> 

</table></td> 

</tr> 

</table> 

</div> 


最 后 ， 通 过 js 文件 中 的 自 定义 函数 artpagination() 将 mr_synopsis_ok.php 文件 中 的 分 页 结果 返回 到 


@. 
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div 标签 synopsis 中 ， 实 现 无 刷新 分 页 的 功能 。js 文件 中 的 关键 代码 如 下 : 


/使 用 XMLHttpRequest 对 象 创建 异步 HTTP 请 求 

function artpagination(url}{ /创建 自 定义 函数 ， 获 取 传 递 的 参数 
xmlHttp.open('get,urltrue); /通过 GET 方 法， 执行 另外 一 个 实现 分 页 功能 的 文件 
xmlHttp.onreadystatechange = function(X{ 

if(xmlHttp.readystate == 4 && xmlHttp.status == 200){ /将 结果 返回 到 div 标签 synopsis 中 
document.getElementByld("synopsis").innerHTML = xmlHttp.responseText; 

} 


. 
xmlHttp.send(null); 
} 


人 
说 明 在 mr_synopsis_ok.php 文件 中 完成 的 分 页 功能 与 mr_synopsis.php 文件 中 是 相同 的 ， 代 码 
不 再 重复 。 


6.5 标准 分 页 
6.5.1 PHP 标准 分 页 功能 概述 


PHP 标准 分 页 在 当前 页 面 中 显示 指定 的 记录 ， 同 时 以 10 页 为 一 个 单位 ， 实 现 上 10 页 和 下 10 页 的 
跳 转 ， 并 且 实 现 当 前 页 中 显示 的 10 个 超 链 接 的 任意 跳 转 。 运 行 结果 如 图 6.10 所 示 。 
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图 6.10 PHP 标准 分 页 运行 结果 
6.5.2 PHP 标准 分 页 功能 实现 过 程 


在 PHP 标准 分 页 中 ,每 页 显示 5 条 数据 。 分 页 跳 转 可 以 分 为 3 个 部 分 : 第 1 部 分 , 根据 查询 结果 ， 
通过 while 语句 在 当前 页 面 中 显示 5 条 记录 信息 ; 第 2 部 分 ， 以 10 页 为 一 个 单位 ， 实 现 上 10 页 和 下 


时 E>) 
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10 页 的 跳 转 ; 第 3 部 分 , 在 当前 页 中 , 在 显示 的 10 个 超 链接 中 任意 跳 转 。 PHP 标准 分 页 由 mr_book.php 
文件 来 完成 ， 其 具体 操作 步骤 如 下 : 

(1) 连接 数据 库 ， 定 义 分 页 中 要 使 用 的 变量 。 包 括 数据 库 的 连接 文件 conn.php、 当 前 页 值 的 变量 
S$page 和 当前 10 页 值 的 变量 $link_type。 关 键 代码 如 下 : 


<?php 
include_once("conn/conn.php"); 
if(isset($_POST[page'])X{ 
$page=$_POST['page'; 

}else if(isset($_GET[page'])X{ 
$page=$_GET['page'; 

jelsef 

$page=1; 


} 

if (isset($_GET[Iink_type])){ 
S$link_type=$_GETT'link_type']; 
}else{ 

S$link_type=0; 


下 

if (isset($_GET[vv])){ 
$vv=$_GET[vv]; 
}else{ 

$vv=0; 


小 
include("function.php"); 
?> 


-/ 

本 2 明 上 述 的 这 段 代码 存储 于 index.php 文件 中 ， 因 为 本 模块 应 用 switch 语句 完成 网 页 的 框架 操 
作 ， 所 有 功能 都 包含 于 index.php 文 件 中 ， 所 以 将 上 述 的 这 段 代码 存储 于 index.php 文件 中 ， 作 为 公 
共 内 容 。 

(2) 在 mr_book.php 文件 中 ， 从 数据 库 中 读 取 数 据 , 统计 出 总 的 记录 数 ,设置 每 页 显示 的 记录 数 ， 
计算 出 共有 几 页 ， 并 通过 while 语句 循环 输出 数据 库 中 的 数据 。 代 码 如 下 : 


<?php 

if($pageX{ 1/ 判断 当前 页 变量 的 值 是 否 存在 
$page_size=5; /定义 每 页 输出 的 记录 数 

$query="select * from tb_forum_send "; // 编 写 查 询 语 句 ， 应 用 count 统计 总 的 记录 数 
$result=$conn->query($query); /执行 查询 语句 


$res=$result->fetchAll(PDO::FETCH_ASSOC); 

$message_count=count($res); 

$page_count=ceil($message_count/$page_size); 。 // 计 算出 总 共有 几 页 
$offset=($page-1)*$page_size; // 输 出 上 一 页 结束 的 记录 数 
$query_2=$conn->query("select * from tb_forum_send limit $offset, $page_size"); 
foreach($query_2 as $value){ 

J 


<tr> 


@. 
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<td width="35%" align="center" bgcolor="#FFFFFF"> 

<a href="index_ok.php?send_id=<?php echo $value[tb_send id];?>" target="_blank"> 

<?php echo $value['tb_send_subject];?></a> 

</td> 

<td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value[tb_send_date'];?></td> 
<td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value[tb_send_user];?></td> 
<td width="10%" align="center" bgcolor="#FFFFFF"> 

<?php 

$query_s=$conn->query("select * from tb_forum_restore "); 

echo count($query_s); 

?> 

</td> 

</tr> 

<?php 

} 

3 


?> 


(3) 创建 分 页 超 链接 , 实现 分 页 中 以 10 页 为 单位 的 跳 转 , 以 及 当前 10 页 中 不 同 页 面 之 间 的 跳 转 。 
关键 代码 如 下 : 


<table width="800" height="30" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor= 
"#DODODO"> 

<tr> 

<td height="22" align="center" bgcolor="#FFFFFF"> 

<span class="STYLE2"> 帖 子 统 计 : <?php echo $message_count;?> 条 &nbsp; </span></td> 

<td align="center" bgcolor="#FFFFFF"> 每 页 显示 : <?php echo $page_size; ?> 条 </td> 

<td align="center" bgcolor="#FFFFFF"> 共 有 : <?php echo $page_count;?> 页 </td> 

<td align="center" bgcolor="#FFFFFF"><span class="STYLE2"> 分 页 : 


<?php 

$next=$link_type*10; /以 10 页 为 一 个 单位 

$n=qlink type-1; 1/ 上 一 个 10 页 的 值 

$m=$link_type+1; /下 一 个 10 页 的 值 

$prev_page=$page-10; // 当 前 页 的 上 10 页 的 值 

$mm="; 

if($link_type==0){ // 判 断 变 量 的 值 是 否 等 于 0， 如 果 等 于 0， 则 输出 首页 图 标 ， 无 超 链接 

echo "<img src=\"images/02.jpg\" width=\"8\" height=\"9\" border=\"OV" title=\" 首 页 \">"; 

}elsef // 和 否则， 为 首页 图 标 设置 超 链接 ， 连 接 到 首页 ， 并 为 上 10 页 的 图 标 设置 超 链接 


echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=0&link_type=0&page=1'> 

<img src=\"images/02.jpg\" width=\"8\" border=\"O\" height=\"9\" border=\"O\" title=\" 首 页 \"></a>&nbsp;"; 
$ccc=$vv-10; // 定 义 变量 ， 变 量 $vv 是 当前 10 页 的 起 始 位 置 ，$ccc 是 上 一 个 10 页 的 起 始 位 置 
echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=$ccc&link_type=$n&page=$prev_page'> 

<img src=\"images/01.jpg\" width=\"8\" height=\"9\" border=\"OV" title=\" 上 10 页 \"></a>"; 

<?php 

// 通 过 for 循环 语句 ， 输 出 当前 页 中 的 10 个 超 链接 

for($j=1;$j<=$page_count;$j++){ ”// 根 据 统计 的 总 的 记录 数 循 环 输出 

S$pnext=$next+$j; // 定 义 当前 页 中 输出 的 页 码 值 
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if($mm==10){ // 判 断 当 $mm 的 值 等 于 10 时 跳出 循环 
break; 

由 

if($mm>$page_count){ /判断 当 $mm 的 值 大 于 总 记录 时 跳出 循环 
break; 


} 

if($page_count-$vv<10X{ 

if($mm>=$page_count-$vv){ // 判 断 当 $mm 的 值 大 于 等 于 总 记录 减 去 当前 10 页 的 起 始 值 时 跳出 循环 
break; 


} 


} 
echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=S$vv&link_type=Slink_type&page=$pnext>"; 


if($page==$pnext){ /为 当前 点 击 的 页 码 描 红 

echo "<span class='STYLE1'> $pnext </span>"; 

}else{ 

echo " $pnext "; /| 输出 当前 10 页 超 链接 中 的 数据 

} 

echo "</a>"; 

$mm=$mm+1; 

} 

?> 

</span><span class="STYLE2"> 

<?php 

$=$vv+$mm; // 定 义 变量 的 值 ， 定 义 下 10 页 的 起 始 位 置 
if($page_count-$vv<=0X{ /判断 当 总 记录 数 小 于 等 于 下 10 页 的 起 始 位 置 时 ， 输 出 尾 页 的 图 标 
echo "<img src=\"images/03.jpg\" width=\"8\" height=\"9\" border=\"Q\" title=\" 尾 页 \">"; 
}else{ // 否 则 输出 下 10 页 的 超 链接 


echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=$vv&link_type=$m&page=$pnext> 
<img src=\"images/03.jpg\" width=\"8\" height=\"g\" border=\"O\" title=\" 下 十 页 %></a>"; 


re 1/ 判断 当 总 记录 数 等 于 0 时 ， 输 出 没有 记录 
echo "没有 记录 !"; 

} 

?> 

</span>&nbsp;&nbsp;</td> 

</tr> 

</table> 


至 此 ， 有 关 PHP 标准 分 页 的 功能 介绍 完毕 ， 详 细 代 码 可 以 参考 本 章光 盘 中 的 内 容 。 
6.6 跳 转 分 页 


6.6.1 PHP 跳 转 分 页 功能 概述 


PHP 跳 转 分 页 的 特点 是 能 够 实现 当前 页 面 到 其 他 指定 页 面 的 跳 转 ， 多 数 情况 下 它 与 标准 分 页 功能 
结合 使 用 ， 以 增加 分 页 操作 的 灵活 性 。 运 行 结果 如 图 6.11 所 示 。 
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图 6.11 PHP 跳 转 分 页 的 运行 结果 
6.6.2 PHP 跳 转 分 页 功能 的 实现 过 程 


在 PHP 跳 转 分 页 中 ， 每 页 显示 5 条 数据 。 分 页 跳 转 可 以 分 为 3 个 部 分 : 第 1 部 分 ， 通 过 文本 框 实 
现 指定 页 面 的 跳 转 功能 ;第 2 部 分 ， 以 10 页 为 一 个 单位 ， 实 现 上 10 页 和 下 10 页 的 跳 转 ; 第 3 部 分 ,在 
当前 页 中 ， 在 显示 的 10 个 超 链接 中 任意 跳 转 。PHP 跳 转 分 页 由 mr_forum.php 文件 来 完成 ， 其 具体 操 
作 步 又 如 下 : 

(1) 连接 数据 库 ， 定 义 分 页 中 要 使 用 的 变量 ， 包 括 数据 库 的 连接 文件 conn.php， 当 前 页 值 的 变量 
$page 和 当前 10 页 值 的 变量 $link_type。 关 键 代码 如 下 : 


<?php 
include_once("conn/conn.php"); 
if(isset($_POST[page])X{ 
$page=$_POST[page]; 

}else if(isset($_GET[page'])X{ 
$page=$_GET[page]; 

}else{ 

$page=1; 


} 

if (isset($_GETIink_type]){ 
Slink_type=$_GET[link_type']; 
Jelse{ 

S$link_type=0; 


» 

if (isset($_GET[vv]){ 
$vw=$_GET[vv]; 
}else{ 


$vv=0; 


} 
include("function.php"); 
?> 
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和 说明 上 述 的 这 段 代码 存储 于 index.php 文件 中 ， 因 为 本 模块 应 用 switch 语句 完成 网 页 的 框架 操 
作 ， 所 有 的 功能 都 包含 于 index.php 文件 中 ， 所 以 将 上 述 的 这 段 代 码 存 储 于 index.php 文件 中 ， 作 为 
公共 内 容 。 
(2) 在 mr_forum.php 文件 中 ， 从 数据 库 中 读 取 数据 ， 统 计 出 总 的 记录 数 ， 设 置 每 页 显示 的 记录 
数 ， 计 算出 共有 几 页 ， 并 通过 while 语句 循环 输出 数据 库 中 的 数据 。 代 码 如 下 : 


<?php 

if($pageX{ /判断 当前 页 变量 的 值 是 否 存 在 
$page_size=5; /定义 每 页 输出 的 记录 数 

$query="select * from tb forum_send "; /| 编写 查询 语句 ， 应 用 count 统计 总 的 记录 数 
$result=$conn->query($query); /| 执行 查询 语句 


$res=$result->fetchAll(PDO::FETCH_ASSOC); 

$message_count=count($res); 

$page_count=ceil($message_count/$page_size); // 计 算出 总 共有 几 页 
$offset=($page-1)*$page_size; /| 输出 上 一 页 结束 的 记录 数 
$query_2=$conn->query("select * from tb_forum_send limit $offset, $page_size"); 
foreach($query_2 as $value){ 

Ss 


<tr> 

<td width="35%" align="center" bgcolor="#FFFFFF"> 

<a href="index_ok.php?send_id=<?php echo $value['tb_send_id'];?>" target="_blank"> 

<?php echo $value['tb_send_subject];?></a> 

</td> 

<td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value[tb_send_date];?></td> 
<td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value[tb_send_user];?></td> 
<td width="10%" align="center" bgcolor="#FFFFFF"> 

<?php 

$query_s=$conn->query("select * from tb_forum_restore "); 

echo count($query_s); 

?> 

</td> 

</tr> 

<?php 

} 

> 


光环 
(3) 创建 一 个 表单 ， 用 于 提交 跳 转 的 页 码 值 ， 并 且 通 过 JavaScript 脚本 判断 提交 的 值 是 否 有 效 。 
代码 如 下 : 
<form name="form1" method="post" action="index.php?lImbs= 明 日 论坛 "onSubmit="return chk()"> 
<input type="submit" name="Submit" value=" 跳 转 ">&nbsp; 
<input name="page" type="text" size="3"> 


<input type="hidden" name="pages" value="<?php echo $page_count;?>"> 
</form> 


@ 
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判断 提交 的 页 码 值 是 否 有 效 是 通过 自 定义 函数 chk() 完 成 的 ， 其 关键 代码 如 下 : 


<script language="javascript"> 

function chk(X{ 

if(form1.page.value<=0 || parselnt(form1.page.value)>parselnt(form1.pages.value)\{ 
alert(" 您 输入 的 页 码 无 效 申 ); 

form1.page.focus(); 

return(false); 

} 

return(true); 

} 


</script> 


(4) 编写 PHP 语句 ， 实 现 分 页 中 以 10 页 为 单位 的 跳 转 ， 以 及 当前 10 页 中 不 同 页 面 之 间 的 跳 转 。 


关键 代码 如 下 : 
<td align="center" bgcolor="#FFFFFF"><span class="STYLE2"> 分 页 : 
<?php 
$next=$link_type*10; /以 10 页 为 一 个 单位 
04 $n=$link_type-1; // 上 一 个 10 页 的 值 
$m=$link_type+1; /下 一 个 10 页 的 值 
$prev_page=$page-10; // 当 前 页 的 上 10 页 的 值 
$mm="; 
if($link_type==0){ // 判 断 变 量 的 值 是 否 等 于 0， 如 果 等 于 0， 则 输出 首页 图 标 ， 无 超 链接 
echo "<img src=\"images/02.jpg\" width=\"8\" height=\"9\" border=\"O\" title=\" 首 页 \">"; 
}else{ 1/ 否则， 为 首页 图 标 设置 超 链接 ， 连 接 到 首页 ， 并 为 上 10 页 的 图 标 设置 超 链接 


echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=0&link_type=0&page=1'> 

<img src=\"images/02.jpg\" width=\"8\" border=\"0\" height=\"9\" border=\"O\" title=\" 首 页 \"></a>&nbsp;"; 
$ccc=$vv-10; // 定 义 变量 ， 变 量 $vv 是 当前 10 页 的 起 始 位 置 ，$ccc 是 上 一 个 10 页 的 起 始 位 置 
echo "<a href='index.php?Imbs= 阴 日 论坛 &vv=S$ccc&link_type=$n&page=$prev_page'> 

<img src=\"images/01.jpg\" width=\"8\" height=\"9\ border=\"O\" title=\" 上 10 页 \"></a>"; 

<?php 

// 通 过 for 循环 语句 ， 输 出 当前 页 中 的 10 个 超 链 接 

for($j=1;$j<=$page_count;$j++) { // 根 据 统计 的 总 的 记录 数 循环 输出 

S$pnext=$next+$j; // 定 义 当前 页 中 输出 的 页 码 值 

if($mm==10){ // 判 断 当 $mm 的 值 等 于 10 时 跳出 循环 


break; 


} 
if($mm>$page_count){ // 判 断 当 $mm 的 值 大 于 总 记录 时 跳出 循环 
break; 


} 

if($page_count-$vv<10X{ 

if($mm>=$page_count-$vv}{ // 判 断 当 $mm 的 值 大 于 等 于 总 记录 减 去 当前 10 页 的 起 始 值 时 跳出 循环 
break; 


} 


} 
echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=$vv&link_type=S$link_type&page=$pnext>"; 
if($page==$pnext){ a 


) PHP 典型 模块 开发 全 程 实录 


echo "<span class='STYLE1> $pnext </span>"; 


}else{ 

echo " $pnext "; /| 输出 当前 10 页 超 链接 中 的 数据 

2 

echo "</a>"; 

$mm=$mm+1; 

} 

?> 

</span><span class="STYLE2"> 

<?php 

$w=$vw+$mm; /定义 变量 的 值 ， 定 义 下 10 页 的 起 始 位 置 
if($page_count-$vv<=0X{ /| 判断 当 总 记录 数 小 于 等 于 下 10 页 的 起 始 位 置 时 ， 输 出 尾 页 的 图 标 
echo "<img src=\"images/03.jpg\" width=\"8\" height=\"9\" border=\"OV" title=\" 尾 页 \">"; 
Jelse{ /否则 输出 下 10 页 的 超 链接 


echo "<a href='index.php?Imbs= 了 明日 论坛 &vv=$vv&link_type=$m&page=$pnext> 
<img src=\"images/03.jpg\" width=\"8\" height=\"9\ border=\"O\" title=\" 下 十 页 ></a>"; 


} 

if($message_count==0){ 1/ 判断 当 总 记录 数 等 于 0 了 时， 输出 没有 记录 
echo "没有 记录 !"; 

} 

Tw 

</span>&nbsp;&nbsp;</td> 


至 此 ， 有 关 PHP 跳 转 分 页 的 功能 介绍 完毕 ， 详 细 代码 可 以 参考 本 章光 盘 中 的 内 容 。 
67 理 - 下 分 页 


6.7.1 PHP 上 下 分 页 功能 概述 


PHP 上 下 分 页 功能 以 当前 页 码 为 基础 ， 实 现 上 一 页 和 下 一 页 的 跳 转 ， 并 且 输 出 上 一 页 中 最 后 一 条 
数据 和 下 一 页 中 第 一 条 数据 的 内 容 。 运 行 结果 如 图 6.12 所 示 。 


fT 明日 科技 司 商 为 时而 从 加 入内 构 姑 


| auud AoGE 有 
相 sume me eam | | 


匾 程 间 典 明日 图 李 精 帖 贷 析 版 权 声明 则 日 简 介 


Duns mma 加 
1 


入村 本 中 必 ， 交 全 hd， 珊 上 中 天 面 呈 
ES 


bg 


图 6.12 PHP 上 下 分 页 的 运行 结果 
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6.7.2 PHP 上 下 分 页 功能 的 实现 过 程 


PHP 上 下 分 页 ， 每 页 显示 2 条 数据 。 分 页 跳 转 可 以 分 成 两 个 部 分 : 第 1 部 分 是 上 下 翻 页 的 跳 转 ， 第 2 部 
分 则 是 以 10 页 为 一 个 单位 ， 实 现 上 10 页 和 下 10 页 的 跳 转 ， 以 及 在 当前 页 显示 的 10 个 超 链接 中 的 任意 跳 
转 。 有 关 第 2 部 分 的 内 容 已 经 在 第 6.6.2 节 中 做 了 详细 介绍 ， 这 里 不 再 袭 述 。 

这 里 主要 介绍 上 下 翻 页 的 跳 转 ， 以 及 输出 上 一 页 的 最 后 一 条 数据 和 下 一 页 的 第 一 条 数据 的 方法 ， 
还 有 分 栏 功能 的 实现 方法 。 其 操作 在 mr_dictionary.php 文件 中 完成 。 

在 输出 数据 库 中 的 数据 时 ， 应 用 的 是 分 栏 方法 ,创建 一 个 1 行 2 列 的 表格 来 输出 数据 库 中 的 数据 。 
实现 的 原理 如 下 : 

(1) 在 while 语句 之 外 ， 定 义 一 个 变量 $number=0。 

(2) 在 while 语句 之 中 ， 通 过 f 语 句 判断 取 模 〈$number % 2) 的 值 是 否 等 于 0， 如 果 等 于 0， 则 
是 行 的 开始 (执行 <tr>)。 

(3) 在 while 语句 之 中 ， 通 过 让 语句 判断 取 模 $number % 2) 的 值 是 否 等 于 0， 如 果 不 等 于 0， 
则 是 行 的 结束 (执行 <tr>〉。 

(4) 在 while 语句 之 外 ， 执 行 Snumber++。 

关键 代码 如 下 : 

<?php 

$number=0; // 定 义 变量 

// 省 略 了 部 分 PHP 代码 


$query_2=mysql_query("select * from tb_mr_bccd limit $offset, $page_size"); 
while($myrow_2=mysql_fetch_array($query_2)X /循环 输出 


if (( $number % 2 ) == OX 1/ 判断 取 模 的 值 是 否 等 于 0 

了 

<tr> <!-- 当 取 模 的 值 等 于 0 时 执行 此 操作 --> 
<?php } ?> 

<td colspan="2" align="center"></td> <!-- 输 出 数据 库 中 的 数据 --> 

<?php 

if(($number%2 ) != 0){ // 判 断 当 取 模 的 值 不 等 于 0 时 

We 

</tr> <!-- 当 取 模 的 值 不 等 于 0 时 执行 此 操作 --> 
<?php } // 结 束 while 循环 

$number++; /变量 值 增加 

?> 


上 一 页 、 下 一 页 跳 转 功 能 的 实现 ， 关 键 就 是 对 当前 页 值 page 的 判断 。 首 先 设置 上 一 页 的 超 链接 和 
获取 上 一 页 中 的 最 后 一 条 数据 ， 关 键 代码 如 下 : 


<td width="205" align="center”bgcolor="#FFFFFF"> 上 一 部 : <a href="index.php?Imbs= 阴 日 编程 词典 &page= 
<?php 

if($page=="" or $page<=1){ 

echo 1; 

}else{ 

echo ceil(($page*2-2)/2); 

} 


We) 
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<?php 
$page_3=($page-2)*2+1; 
if($page_3<=0){ 


$page_3=0; 

$query_3="select* from tb_mr_bccd limit $page_3,1"; 
$result3=$conn->prepare($query_3); // 准 备查 询 语句 
$result3->execute(); /执行 查询 语句 ， 并 返回 结果 集 


$res3=$result3->fetch(PDO::FETCH_ASSOC); 
echo $res3[tb_bccd_name]; 
?></a></td> 


然后 获取 下 一 页 超 链 接 中 page 的 值 和 下 一 页 中 的 第 一 条 数据 ， 关 键 代码 如 下 : 


<td width="238"” align="center”bgcolor="#FFFFFF"> 下 一 部 : <a href="index.php?Imbs= 阴 日 编程 词典 
&page=<?php 

if($page=="" or $page<=1)X{ 
echo 2; 

}else if($page>=$page_count){ 
echo $page_count; 

}else{ 

echo ceil(($page*2+1)/2); 
}?>"> 

<?php 

$page_4=($page)’2; 
if($page_4<=0){ 

$page_4=2; 


} 
if($page_4==$page_count*2){ 
$page_4=$page_count*2-1; 


$query_4="select * from tb_mr_bccd limit $page_4,1"; 
$result4=$conn->prepare($query_4); // 准 备查 询 语句 
$result4->execute(); /执行 查询 语句 ， 并 返回 结果 集 
$res4=$result4->fetch(PDO::FETCH_ASSOC); 

echo $res4[tb_bccd_name]; 

?></a></td> 


PHP 上 下 分 页 中 的 其 他 内 容 已 经 在 第 6.6.2 节 中 介绍 过 了 ,这 里 不 再 袭 述 ,完整 程序 代码 请 参考 本 
章光 得 中 的 内 容 。 


6.8 技术 提炼 


6.8.1 ” 自 定义 函数 替换 超 长 文本 中 的 特殊 字符 技术 


在 输出 超 长 文本 中 的 数据 时 ， 应 用 自 定义 函数 unhtml0 实 现 对 超 长 文本 中 的 特殊 字符 进行 替换 。 
unhtml(0 函 数 的 语法 如 下 : 


@ 
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function unhtml($content)f // 定 义 自 定义 函数 的 名 称 
$content=htmlspecialchars($content); // 转 换文 本 中 的 特殊 字符 
S$content=str_replace(chr(13),"<br>",$content); // 蔡 换文 本 中 的 换行 符 
S$content=str_replace(chr(32),"&nbsp;", $content); // 共 换文 本 中 的 &nbsp; 
$content=str_replace("[_[","<", $content]; // 蔡 换文 本 中 的 大 于 号 
$content=str_replace(") )",">",$content); // 蔡 换文 本 中 的 小 于 号 
$content=str_replace("|_|"," ",$content); // 蔡 换文 本 中 的 空格 
return trim($content); /删除 文本 中 首尾 的 空格 


上 


首先 ， 通过 file_get_contents0) 函 数 读 取 超 长 文本 中 的 数据 ,返回 一 个 字符 串 变量 $counter，unhtml0 
函数 就 是 对 这 个 字符 串 进 行 操作 ， 蔡 换 字符 串 中 的 特殊 字符 ， 然 后 再 执行 其 他 操作 。 关 键 代 码 如 下 : 


<?php 

include("function.php"); // 调 用 指定 的 文件 
if($pageX{ 

$counter=file_get_contents("file/file .txt"); // 读 取 指 定 文件 的 内 容 
$length=strlen(unhtml($counten)); // 获 取 字 符 串 长 度 
$page_count=ceil($length/1050); /| 统计 每 页 的 字符 数 
$c=msubstr($counter,0,($page-1)*1050); // 计 算 上 页 
$c1=msubstr($counter,0,$page*1050); // 计 算 当 前 页 

echo substr($c1,strlen($c),strlen($c1)-strlen($c)); // 输 出 当前 页 的 内 容 
} 

?> 


file_get_contents() 函 数 将 指定 文件 中 的 内 容 读 入 到 一 个 字符 串 。 如 果 操 作 系 统 支持 内 存 映射 技术 ， 
还 会 应 用 内 存 映 射 技术 来 增强 其 读 入 文件 的 性 能 。 该 函数 的 语法 如 下 : 


String file_get_contents(string filename [, int use_include_path [, resource context]) 


file_get_contentsO) 函 数 的 参数 说 明 如 表 6.1 所 示 。 
表 6.1 file_get_contents() 函 数 的 参数 说 明 
说 了 明 
必要 参数 。 用 于 指定 某 文件 的 路 径 及 名 称 
可 选 参数 。 如 果 将 该 参数 设置 为 tue，PHP 会 尝试 按照 include_path 标准 包含 路 径 中 的 每 个 指向 
去 打开 文件 
可 选 参数 。 设 置 提高 文件 流 性 能 的 一 些 选项 


6.8.2 自 定 义 函 数控 制 超 长 文本 的 输出 技术 


PHP 超 长 文本 分 页 显示 ， 控 制 超 长 文本 中 的 数据 输出 应 用 的 是 自 定义 函数 msubstr()， 该 函数 的 语 
法 如 下 : 


function msubstr($str,$start,$lenX 。“”//$str 指 的 是 字符 串 ，$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 


S$tmpstr="™"; 
轧 


参数 


filename, 


use_include_path 


context 
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$strlen=$start+$len; // 用 $strien 存储 字符 串 的 总 长 度 〈 从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 
for($i=0;$i<$strlen;$i++){ /通过 for 循环 语句 循环 读 取 字 符 串 

if(ord(substr($str, $i,1))>0xaOX{ // 如 果 字符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 0xa0， 则 表示 为 汉字 
$tmpstr.=substr($str,$i,2); /| 每 次 取出 两 位 字符 赋 给 变量 $tmpstr， 即 等 于 一 个 汉字 

$i++; /变量 自 加 1 

Jelse{ // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
S$tmpstr.=substr($str,$i,1);} 

} 

return $tmpstr; /| 输出 字符 串 

} 


首先 ， 应 用 include 调用 指定 的 包含 文件 function.php; 然后 ， 应 用 file_get_contents() 函 数 读 取 指 定 
文本 文件 中 的 数据 ， 并 应 用 自 定义 函数 unhtml0 蔡 换 字符 串 中 的 特殊 字符 ， 最 后 ， 应 用 msubstr() 函 数 
控制 数据 的 输出 。 关 键 代 码 如 下 : 


<?php 

include("function.php"); 

if($pageX{ 

$counter=file_get_contents("file/file.txt"); // 读 取 文 本 文件 中 的 数据 ， 返 回 一 个 字符 串 
$length=strlen(unhtml($counten)); // 计 算 字符 串 的 长 度 ， 蔡 换 特 殊 字 符 
$page_count=ceil($length/1050); 1/ 对 字符 串 进行 分 页 处 理 
$c=msubstr($counter,0,($page-1)*1050); // 通 过 自 定义 函数 获取 上 一 页 中 的 数据 
$c1=msubstr($counter,0,$page*1050); // 通 过 自 定义 函数 获取 当前 页 中 的 数据 
echo substr($c1,strlen($c),strlen($c1)-strlen($c)); // 获 取 当 前 页 中 要 输出 的 数据 

四 


6.8.3 ”连接 数据 库 类 的 创建 


要 实现 数据 库 中 数据 的 分 页 显示 ， 首 先 必 须 与 数据 库 建 立 连 接 ， 这 里 通过 一 个 自 定义 的 类 来 
实现 数据 库 的 连接 ， 其 中 操作 数据 库 使 用 PDO 技术 。 该 类 存储 于 06\01\system\system.class.inc.php 
文件 中 ， 其 代码 如 下 : 


class ConnDB{ /数据 库 连 接 类 
var $dbtype; 

var $host; 

var $user; 

var $pwd; 

var $dbname; 

function ConnDB($dbtype, $host, $user,$pwd,$dbname){ /构造 方法 
$this->dbtype=$dbtype; 

S$this->host=$host; 

S$this->user=$user; 

$this->pwd=$pwd; 

$this->dbname=$dbname; 


} 


@. 
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function GetConnld(X{ /实现 数据 库 的 连接 并 返回 连接 对 象 
if($this->dbtype=="mysql" || Sthis->dbtype=="mssql")f /判断 数据 库 类 型 
$dsn="$this->dbtype:host=$this->host;dbname=$this->dbname"; 。“// 定 义 连接 语句 

}else{ 

$dsn="$this->dbtype:dbname=$this->dbname"; 

; 

try{ 

$conn = new PDO($dsn, $this->user, $this->pwd); /| 初始 化 一 个 PDO 对 象 ， 创 建 数据 库 连 接 对 象 $gconn 
$conn->query("set names utf8"); /设置 编码 格式 

return $conn; // 返 回 连接 对 象 


} catch (PDOException $e){ 

die ("Error!: " . $e->getMessage() . "<br/>"); 
} 

} 


6.8.4 ”操作 数据 库 类 的 创建 


数据 库 连 接 成 功 后 ， 接 下 来 进行 数据 库 的 操作 ， 这 里 同样 对 PDO 操作 数据 库 的 方法 进行 了 封装 ， 


在 AdminDB 类 中 ， 可 以 执行 数据 的 添加 、 更 新 、 删 除 和 查询 操作 。 类 文件 存储 于 06\01\system\system. 
class.inc.php 中 ， 其 代码 如 下 : 


class AdminDB{ /数据 库 管 理 类 

function ExecSQL($sqlstr, $connX{ 

$sqltype=strtolower(substr(trim($sqlstr),0,6)); 

$rs=$conn->prepare($sqlstr); 1// 准 备查 询 语句 
$rs->execute(); /执行 查询 语句 ， 并 返回 结果 集 
if($sqltype=="select"){ 

$array=$rs->fetchAll(PDO::FETCH_ASSOC); // 获 取 结 果 集中 的 所 有 数据 
if(count($array)==0 || $rs==false) 

return false; 

else 

return $array; 

}elseif ($sqltype=="update" || $sqltype=="insert" || $sqltype=="delete"X{ 

if($rs) 

return true; 

else 

return false; 

} 

} 

3 


6.8.5 分 页 类 的 创建 


为 了 让 分 页 技术 的 应 用 更 加 简单 ， 可 以 将 分 页 的 方法 写 入 一 个 类 中 ， 这 样 就 可 以 在 需要 使 用 分 页 
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的 地 方 直接 调用 这 个 类 实现 分 页 的 功能 。 分 页 类 同样 存储 于 06\01\system\system.class.inc.php 文件 中 ， 
其 代码 如 下 : 


class SepPage{ 

Var $rs; 

var $pagesize; 

var $nowpage; 

var $array; 

var $conn; 

var $sqlstr; 

function ShowData($sqlstr,$conn,$pagesize,$nowpage){ 
ifllisset($nowpage) || $nowpage=="") 
S$this->nowpage=1; 

else 

S$this->nowpage=$nowpage; 
S$this->pagesize=$pagesize; 

S$this->conn=$conn; 

S$this->sqlstr=$sqlstr; 
S$offset=($this->nowpage-1)*$this->pagesize; 
$sql=$this->sqlstr." limit $offset, $this->pagesize”; 
$result=$this->conn->prepare($sql); 
$result->execute(); 
$this->array=$result->fetchAll(PDO::FETCH_ASSOC); 
if(count($this->array)==0 || $this->array==false) 
return false; 

else 

return $this->array; 


3 


/定义 方法 
// 判 断 变量 值 是 否 为 空 
// 定 义 每 页 的 起 始 页 


// 定 义 每 页 输出 的 记录 数 
/I 连接 数据 库 返 回 的 标识 
/| 执行 的 查询 语句 


// 准 备查 询 语句 
/| 执行 查询 语句 ， 并 返回 结果 集 
// 获 取 结 果 集 中 的 所 有 数据 


function ShowPage($contentname, $utits, $anothersearchstr,$anothersearchstrs,$class}{ 


$str=""; 

S$res=$this->conn->prepare($this->sqlstr); 
$res->execute(); 
$this->array=$res->fetchAll(PDO::FETCH_ASSOC); 
$record=count($this->array); 
$pagecount=ceil($record/$this->pagesize); 


/准备 查询 语句 

// 执 行 查询 语句 ， 并 返回 结果 集 
/获取 结果 集中 的 所 有 数据 

/| 统计 记录 总 数 

1/ 计算 共有 几 页 


$str.=$contentname."&nbsp;".$record."&nbsp:;".$utits."&nbsp; 每 页 &nbsp; 
".$this->pagesize."&nbsp;".$utits."&nbsp; 第 &nbsp;".$this->nowpage."&nbsp; 页 / 共 &nbsp;".$pagecount."&nbsp; 页 "; 


$str.="&nbsp;&nbsp;&nbsp;&nbsp;"; 
if($this->nowpage!=1) 
$str.="<a href=".$_SERVER[PHP_SELF']." 


?page=1&page_type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 首 页 </a>"; 


else 

$str.="<font color=#555555'> 首 页 </font>"; 
$str.="&nbsp;"; 

if($this->nowpage!=1) 

$str.="<a href=".$_SERVER[PHP_SELF]."? 
page=".($this->nowpage-1)." 
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&page_type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 上 一 页 </a>"; 
else 

$str.="<font color=#555555'> 上 一 页 </font>"; 

$str.="&nbsp;"; 

if($this->nowpage!=$pagecount) 

$str.="<a href=".$_SERVER[PHP_SELF']."?page=".($this->nowpage+1)." 
&page_type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 下 一 页 </a>"; 
else 

$str.="<font color=#555555'> 下 一 页 </font>"; 

S$str.="&nbsp;"; 

if($this->nowpage!=$pagecount) 

$str.="<a href=".$_SERVER[PHP_SELF']."?page=".$pagecount.” 
&page_type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 尾 页 </a>"; 
else 

$str.="<font color=#555555'> 尾 页 </font>"; 

if(count($this->array)==0 || $this->array==false) 

return "无 数据 ! "; 

else 

return $str; 

} 

} 


在 这 个 分 页 类 中 ,包含 ShowData($sqlstr,$conn,$pagesize,$nowpage) 和 ShowPage($contentname,$utits, 


$anothersearchstr,$anothersearchstrs,$class) 两 个 方法 。 其 中 ShowData() 方 法 根据 参数 传递 的 SQL 语句 ， 
完成 数据 的 分 页 查询 操作 ， 并 且 返 回 查 询 结果 。 在 ShowPage() 方 法 中 根据 参数 传递 的 值 定义 分 页 超 链 
接 的 输出 内 容 。 


6.8.6 ”类 的 实例 化 


类 创建 成 功 之 后 ， 要 想 在 程序 中 调用 类 中 的 方法 完成 指定 的 操作 ， 必 须 先 完成 类 的 实例 化 操作 ， 


通过 类 实例 化 返回 的 对 象 才 可 以 调用 指定 的 方法 。 有 关 类 的 实例 化 操作 在 本 模块 中 都 统一 存储 于 


06\01\system\system.inc.php 文件 中 ， 其 代码 如 下 : 


<?php 

require("system.smarty.inc.php"); // 包 含 Smarty 配置 类 
require("system.class.inc.php"); // 包 含 数据 库 连 接 和 操作 类 
$connobj=new ConnDB("mysql","localhost","root","111","db_pagination"); // 数 据 库 连接 类 实例 化 
$conn=$connobj->GetConnld(); /执行 连接 操作 ， 返 回 连接 标识 
$admindb=new AdminDB(); /数据 库 操作 类 实例 化 
$seppage=new SepPage(); /分 页 类 实例 化 

$usefun=new UseFun(); /使 用 常用 函数 类 实例 化 
$smarty=new SmartyProject(); ll 调用 Smarty 模板 

function unhtml($paramsj{ 

extract($params); 

$text=$content; 
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global $usefun; 

return $usefun->UnHtml(S$text); 

和 

$smarty->register_ function("unhtml","unhtml"); // 注 册 模 板 函数 
$s 


6.9 本 章 小 结 


本 章 详细 讲解 了 如 何 实现 分 页 程序 流程 。 同时， 在 分 页 显示 的 程序 过 程 中 ， 采 用 了 Ajax 无 刷新 分 
页 技术 ， 使 页 面 没 有 全 部 跳 转 ， 只 局 部 刷新 。 通 过 本 章 的 学 习 ， 读 者 不 仅 可 以 了 解 Ajax 技术 的 使 用 ， 
而 且 可 以 熟悉 分 页 显示 的 开发 思想 。 


人 将 
时 


论坛 模块 


(PHP+MySQL 实现 ) 


论坛 已 经 成 为 时 下 一 种 比较 普遍 的 网 上 交流 手段 ， 其 表现 形式 多 
种 多 样 ， 有 个 人 论坛 、 企 业 论 坛 、 图 书 论坛 和 软件 论坛 等 ， 虽 说 名 目 
众多 ， 但 都 是 围绕 着 一 个 “ 论 ” 字 在 进行 ， 其 原理 是 相同 的 ， 都 是 对 
基 个 事物 、 事 情 或 者 人 物 发 表 自 己 的 意见 和 看 法 。 本 章 将 详细 讲解 论 
坛 的 开发 流程 和 在 创建 中 使 用 的 关键 技术 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

H 小 纸 条 信息 无 刷新 输出 
连接 远程 MySQL 数据 库 
树 状 导航 菜单 
帖子 的 发 布 、 回 复 
帖子 置顶 
帖子 引用 、 收 藏 
屏蔽 回帖 


豆 吾 于 于 于 至 
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7.1 论坛 模块 概述 


随 着 互联 网 的 发 展 ， 网 络 信息 资源 也 不 断 的 丰富 ， 而 以 动态 性 和 交互 性 为 特征 的 网 络 论坛 是 当中 
最 丰富 、 最 开放 和 最 自由 的 网 络 信息 资源 ， 是 最 受 欢 迎 的 一 种 信息 交流 方式 。 目 前 实现 论坛 功能 的 开 
发 语言 有 很 多 种 ， 主 要 有 PHP、JSP、ASP 和 ASP.NET， 其 中 PHP 借助 于 开源 的 优势 必 将 成 为 网 络 开 
发 的 必然 趋势 。 

网 络 论坛 和 互联 网 上 的 其 他 信息 一 样 ， 具 有 范围 广 、 内 容 庞 杂 、 动 态 变化 性 强 等 特点 ， 其 中 两 个 
最 重要 的 特点 是 交互 性 和 时 效 性 。 交 互 性 是 指 论坛 用 户 能 够 参加 到 论坛 信息 的 交流 过 程 中 来 ， 可 以 在 
论坛 中 发 布 自己 的 信息 并 且 可 以 得 到 其 他 用 户 的 反馈 ， 这 是 网 络 论坛 信息 最 基本 和 最 重要 的 特征 。 一 
位 用 户 发 布 信息 ， 往 往 有 很 多 人 回应 ， 可 以 开展 多 人 讨论 ， 这 是 一 个 多 向 交流 信息 的 过 程 ， 它 使 人 们 
能 够 便捷 地 进行 交流 。 由 于 互联 网 信息 传播 的 快捷 与 方便 以 及 网 络 论坛 的 交互 性 ， 使 得 人 们 能 够 及 时 
地 从 论坛 中 获取 某 一 技术 和 行业 的 发 展 动态 与 最 新 的 进展 ， 专 业 技术 论坛 中 信息 的 时 效 性 更 强 ， 一 些 
最 新 的 技术 信息 都 可 以 从 论坛 中 获取 。 


7.1.1 系统 功能 结构 


论坛 是 一 个 发 布 帖子 和 回复 帖子 的 过 程 ， 根 据 论 坛 的 基本 功能 ， 整 理 出 论坛 模块 的 功能 结构 图 ， 
如 图 7.1 所 示 。 


帖 | | 所 
会 | | 公 | | 帖 | | 项 | | 子 | | 属 
员 | | 告 | | 子 | | 帖 | | 类 | | 专 
管 | | 管 | | 管 | | 管 | | 别 | | 区 
理 | | 理 | | 理 | | 理 | | 管 | | 管 
理 | | 理 


7.1 论坛 模块 的 功能 结构 图 
根据 功能 结构 图 中 描述 的 功能 ， 设 计 了 一 个 完整 的 论坛 模块 的 开发 流程 ， 如 图 7.2 所 示 。 
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Tb_ mail _ box (信箱 ) 


Tb_ my_collection (收藏 夹 ) 


头像 选择 技术 


Tb forum affiche (公告 7 


Tb _forum big type (专区 ) 


Tb_forum restore (回帖) 


吕 


Tb_forum_send (发 帖 ) 


Tb_forum_small_type (类 别 ) 


Tb_forum user (用 户 ) 


Tb_my_friend (好 友 ) 
| 


图 7.2 论坛 模块 的 开发 流程 


控制 用 户 权限 


置 项 、 引 用 、 屏 蔽 


修改 、 删 除 、 结 帖 


7.1.2 程序 预览 


一 个 完整 的 论坛 系统 是 由 多 个 模块 共同 构建 的 ， 下 面 列 出 一 些 主要 模块 的 运行 结果 ， 让 读者 对 这 
个 论坛 模块 有 一 个 初步 的 了 解 和 认识 。 
论坛 首页 运行 效果 如 图 7.3 所 示 。 其 中 包括 树 状 导航 菜单 、 用 户 登 录 、 初 始 化 信息 等 。 


A MNGRISOFT 
| 话 福 关子 训 


实时 升级 服务 


编程 词 奥 的 用 进 


本 不 入 二 水 和 夺 二 芭 。 宰 学 人 从 在 了 并 全 志江 注 可 上 示人 下 于 略语 
总 芋 和 和 和 个 卉 。 基 及 汉 和 为 扩 人 和 池 寺 万 冯 了 这 时光 中村 训 村 纺 入世 办 的 明 
be EOL 人 宇和 作 了 


过 才 估计， 人 ft 所 生字 玫 御 ， 亿 


地 ， 开 公共 玫 坟 下 | 
二 复生 件 革 机 人 全 备 半 和 慰 ， 


7.3 论坛 首页 运行 效果 
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我 的 信箱 运行 结果 如 图 7.4 所 示 。 其 主要 功能 是 收发 站 内 邮件 、 接 收 系统 信息 、 发 送 好 友 验 证 请 求 
等 操作 ， 并 可 以 在 该 模块 中 对 个 人 收发 的 站 内 邮件 进行 管理 。 


laonsk 您 好 : 您 现在 有 0 条 未 读 信息 ? 


收 驯 往 发 件 箱 写 信 


人: 于 注意: 收 件 人 的 注册 用 户 各 


短信 主题 ”辽东 原 主 申请 
想 申 请 PHP 概 块 的 版 主 ! 司 
短信 内 容 > 
可 
me 
全 MNGRIsoFT 
编程 莫 之 名 


图 7.4 我 的 信箱 运行 结果 


用 户 登 录 论坛 程序 后 ， 可 以 单 击 相关 分 类 中 的 “我 要 提问 ” 超 链接 发 表 新 帖 。 在 发 表 新 帖 的 同时 ， 
该 模块 也 支持 个 人 附件 的 上 传 。 发 表 新 帖 的 运行 结果 如 图 7.5 所 示 。 
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A MINGRISOFT 
”人 久 社 甘 之 名 


图 7.5 发 表 新 帖 的 运行 结果 


在 成 功 发 表 新 帖 后 ， 新 帖 会 被 放置 到 该 类 帖子 浏览 页 面 的 首位 ， 这 样 其 他 用 户 可 以 在 第 一 时 间 浏 
览 到 最 新 的 帖子 内 容 。 其 运行 结果 如 图 7.6 所 示 。 


EE ee ee 
大 | ”在 关闭 和 局 到 县 之 后 应 用 4ate 0 函数 警告 问 是 2011-02-26 10:36:32 | 多 蒙 / 多 蒙 0/2 
帖子 统计 :2 条 分 页 : MI2 网 我 要 提问 


图 7.6 最 新 帖子 发 布 运行 结果 


@ 
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为 了 使 论坛 功能 更 加 合理 、 完 美 ， 本 模块 增加 了 帖子 置顶、 帖子 引用 、 帖 子 收藏 和 屏蔽 帖子 等 特 
殊 功 能 ， 其 中 对 于 发 表 新 帖 的 浏览 和 回复 如 图 7.7 所 示 。 


的 汪 开 发 Ye 
添加 B 我 的 收藏 大 
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习 
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图 7.7 帖子 的 浏览 和 回复 运行 结果 
论坛 后 台 作为 论坛 的 管理 平台 ， 在 整个 管理 体系 中 占据 着 非常 重要 的 位 置 ， 该 模块 主要 用 于 管理 
用 户 的 个 人 资料 以 及 帖子 的 相关 信息 ， 并 可 以 备份 和 恢复 论坛 系统 的 相关 数据 内 容 。 论 坛 后 台 管理 运 
行 效果 如 图 7.8 所 示 。 


AL 
MINGRISOFT 实时 升级 服务 RN 
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图 7.8 论坛 后 台 管 理 运行 效果 
另外 ， 该 论坛 模块 还 包括 一 些 辅助 的 功能 ， 如 我 发 布 的 帖子 和 我 参与 的 帖子 等 。 为 了 便于 论坛 管 
理 ， 增 加 了 管理 员 管理 论坛 的 功能 ， 包 括 发 布 帖子 、 回 复 帖 子 、 帖 子 类 别 和 置顶 帖子 以 及 数据 的 备份 


和 恢复 等 内 容 ， 由 于 篇 幅 有 限 ， 这 里 不 能 一 一 列举 。 
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7.2 ”热点 关键 技术 


在 论坛 模块 的 开发 过 程 中 ， 有 些 关 键 的 技术 是 不 可 缺少 的 ， 
下 面 就 对 论坛 中 用 到 的 关键 技术 进行 详细 介绍 。 了 
7.2.1 树 状 导航 菜单 
+ ”软件 开发 
通过 树 状 导 航 菜单 能 够 对 网 站 中 的 内 容 进行 合理 的 分 类 处 
理 ， 进 而 使 网 站 的 布局 更 加 合理 。 这 里 将 树 状 导航 菜单 的 技术 应 i 
用 到 论坛 中 帖子 类 别 的 处 理 上 ， 改 变 了 以 往 以 整 页 篇 幅 输出 帖子 一 
类 别 的 设计 思路 ， 使 论坛 中 帖子 类 别 的 输出 更 加 合理 、 规 范 。 其 a 
运行 结果 如 图 7.9 所 示 。 ”， 操作 系统 
树 状 导 航 菜 单 的 设计 原理 是 ， 首 先 ， 从 数据 库 中 读 取 论坛 中 


所 属 专区 的 数据 进行 输出 ， 然 后 ， 通 过 JavaScript 脚本 语句 控制 图 7.9 通过 树 状 菜单 输出 帖子 类 别 
单元 格 中 内 容 的 隐藏 和 显示 ;， 最 后 ， 根 据 所 属 的 专区 读 取 对 应 帖 
子 类 别 的 数据 。 其 关键 就 是 单元 格 属性 的 设置 和 JavaScript 脚本 语句 的 应 用 。 
(1) 在 left.php 文件 中 ， 首 先 从 数据 库 的 专区 表 中 读 取 论 坛 中 所 属 专区 的 数据 。 代 码 如 下 : 
<?php $query=mysql_query("select* from tb_forum_big_type"); 
while($myrow=mysql_fetch_array($query))( ”// 循 环 输出 数据 库 中 的 数据 
$querys=mysql_query("select * from tb_forum_small_type where tb_big_type_ content='$myrow 
[tb_big_type_content]' "); /以 所 属 专区 为 条 件 ， 读 取 对 应 专区 中 帖子 的 类 别 的 数据 


$myrows=mysql_fetch_array($querys); 
?> 


(2) 创建 表格 ， 输 出 所 属 专区 的 数据 ， 应 用 onClick 事件 调用 JavaScript 语句 open_close() 控 制 单 
元 格 中 内 容 的 隐藏 和 显示 ， 其 中 还 为 单元 格 中 的 内 容 设置 了 超 链接 ， 链 接 到 指定 的 文件 。 代 码 如 下 : 


<td width="84%" height="24" onClick="javascript:open_close(id_a<?php echo 
$myrow[tb_big_type_id];?>);" > ”< 此 控制 单元 格 的 属性 -> 
<!-- 设 置 超 链接 --> 
<a href="content.php?content=<?php echo S$myrow["tb_big_type_content"];?>&&content_1=<?php echo 
$myrows["tb_small_type_content"];?>" target="contentFrame"><?php echo $myrow["tb_big_type_content"];?>" 
</a> 
</td> 


(3) 创建 表格 ,设置 表格 ID 的 值 和 style 的 值 。 以 帖子 的 所 属 专区 为 查询 条 件 ， 从 类 别 数据 表 中 
读 取 符合 条 件 的 数据 ， 将 读 取 到 的 数据 循环 输出 到 单元 格 中 。 同 样 也 为 单元 格 中 的 内 容 设置 了 超 链 接 ， 
链接 到 指定 的 文件 。 代 码 如 下 : 


@ 
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<table width="170" border="0" cellpadding="0" cellspacing="0" bgcolor="#8394BF" id="id_a<?php echo $myrow 
["tb_big_type_id"];?>" style="display:none"><! 一 设置 表格 的 ID 和 样式 --> 
<?php 
$query_1=mysql_query("select * from tb_forum_small_type where tb_big_type_content='$myrow[tb_ 
big_type_content] "); 
while($myrow_1=mysql_fetch_array($query_1)X 
rs 
<tr> 
<td align="right">&nbsp;</td> 
<td height="23">&nbsp;&nbsp; 
<a href="content.php?content=<?php echo $myrow["tb_big_type_content"];?>&&content_1=<?php 
echo $myrow_1["tb_small_type_content"];?>" target="contentFrame"> 
<?php echo $myrow_1["tb_small_type_content"];?></a></td> 
</tr> 
<?php }?></table> 
<?php }?> 
</td></tr></table> 


其 中 ， 调 用 JavaScript 脚本 的 代码 如 下 : 


<script language="javascript"> 
function open_close(x){ /创建 自 定义 函数 
if(x.style.display=="", // 判 断 当 x 的 值 为 空 时 ， 将 x 的 值 更 改 为 none 
x.style.display="none”"; 
jelse if(x.style.display=="none"){ ”// 否 则 ， 当 x 的 值 等 于 none 时 ， 将 x 的 值 改 为 空 
x.style.display=""; 
} 
} 


</script> 


其 主要 功能 是 实现 单元 格 的 隐藏 和 显示 ， 即 当 单 元 格 处 于 显示 状态 时 ， 单 击 单元 格 则 执行 隐藏 的 
操作 ， 当 单元 格 处 于 隐藏 状态 时 ， 单 击 单元 格 则 执行 显示 的 操作 。 


es 
所 属 专区 ， 而 标识 content 1 则 是 帖子 的 类 别 。 通 过 超 链接 的 target 属性 指定 链接 文件 的 名 称 为 


contentFrame。 


至 此 ， 树 状 导航 菜 单 技术 介绍 完毕 。 有 关 树 状 导 航 菜单 的 具体 应 用 可 以 参考 本 章程 序 中 的 left.php 
文件 。 


7.2.2 ”帖子 置顶 


帖子 置顶 就 是 将 指定 的 帖子 在 所 有 帖子 的 顶端 显示 ， 用 于 突出 帖子 的 特殊 性 。 只 有 管理 员 拥有 帖 
子 置 项 的 权限 ， 其 他 任何 人 都 不 具备 这 个 权限 。 帖 子 置顶 的 操作 过 程 如 图 7.10 所 示 。 
@ 
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发 由 时 间 :2008-09-05 10 19:36 樟 主 疆 耻 
得 PHE6 与 ?HT5 有 什么 不 同 之 处 了 


| 当前 用 户 量 否 是 管理 员 


ET 


un bser=|$_SESSION Itb forun_ vser]” and tb fon 


pn type=" 2 "); 


”where tb. sand_id= $CET[pernute_i 
根据 演变 队 栏 目标 识 ， 更 新 指定 帖子 的 字段 


t》 让 ert (帖子 置顶 记功 | ) ;history, back(] 4/script》"; 


it》 alert (帖子 置顶 失败 |" ) ;history,back()] -script)" 


pt》alert( 忽 不 具备 该 权 限 1 ) ，histocy,back() ;Wascript) 


7.10 ”帖子 置顶 的 操作 过 程 


帖子 置顶 的 操作 原理 是 : 根据 帖子 的 ID 设置 一 个 超 链 接 ， 在 链接 的 permute_send.php 文件 中 实现 
帖子 置顶 的 操作 。 
(1) 在 send_forum_content.php 文件 中 ,创建 一 个 “置顶 ” 超 链 接 ， 链 接 的 标识 为 对 应 帖子 的 ID， 
链接 到 文件 permute_send.php 中 ， 在 该 文件 中 实现 帖子 置顶 的 操作 。 关 键 代码 如 下 : 


<!- 设 置 执行 帖子 置顶 的 超 链接 -> 
<a href="permute_send.php?permute_id=<?php echo $myrow_3["tb_send_id"];?>"> 和 置顶 </a> 


(2) 创建 permute_send.php 文件 ， 实 现 帖 子 置顶 的 操作 。 首 先 ， 判 断 当 前 的 用 户 是 否 是 管理 员 ， 
这 里 用 户 权 限 的 设置 是 通过 用 户 信息 表 (tb_forum_user) 中 tb_forum type 字段 的 值 来 控制 的 ， 如 果 
tb_forum_type 字段 的 值 是 1， 则 代表 注册 的 会 员 ; 如 果 值 是 2， 则 代表 是 管理 员 。 
如 果 当 前 用 户 是 管理 员 , 将 指定 帖子 的 tb_send_type 字段 的 值 更 新 为 1; 否则 不 执行 更 新 数据 的 操 
并 且 弹 出 提示 信息 “您 不 具备 该 权限 !”， 返 回 到 上 一 页 。 相 关 代 码 如 下 : 


作 


<?php 

session_start(); 

include("conn/conn.php"); 

$query=mysql_query("select * from tb_forum_user where tb_forum_user='$_SESSIONItb_forum_user] and 


tb_forum_type="2"); 1/ 判断 当前 用 户 是 否 是 管理 员 
if(mysql_num_rows($query)>0X{ // 当 前 用 户 是 管理 员 ， 执 行 更 新 语句 实现 帖子 置顶 


$query=mysql_query("update tb_forum_send settb_send_type='1' where tb_send _id='$_GET[permute_id]"); 
if($query==trueX{ 
echo "<script> alert(' 帖 子 置顶 成 功 !"); history.back();</script>"; 


jelse{ 
echo "<script> alert(' 帖 子 置 项 失败 !); history.back();</script>"; 
} 
}else{ 
echo "<script> alert( 您 不 具备 该 权限 !"); history.back();</script>"; 
} 
?> 
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帖子 置顶 技术 讲解 完毕 ， 有 关 该 技术 的 完整 应 用 可 以 参考 后 面 小 节 中 的 程序 。 


7.2.3 帖子 引用 


帖子 引用 是 指 在 浏览 帖子 时 ， 针 对 某 个 回复 的 帖子 或 者 自己 的 看 法 与 楼 上 的 看 法 相同 ， 此 时 就 可 
以 单 击 引用 超 链 接 ， 直 接 将 楼 上 回复 的 帖子 进行 引用 ， 作 为 自己 的 回复 帖子 进行 提交 。 帖 子 引用 的 操 
作 流 程 如 图 7.11 所 示 。 
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图 7.11 帖子 引用 的 操作 流程 

帖子 引用 的 实现 原理 是 : 首先， 在 帖子 浏览 的 页 面 中 针对 每 个 回复 的 帖子 设置 一 个 “引用 ” 超 链 
接 , 这 里 将 其 链接 到 本 页 send_forum_content.php 文件 中 ,设置 链接 标识 cite 为 回复 帖子 的 ID， 添 加 销 
点 bottom; 然后 ， 在 指定 输出 回帖 内 容 的 表格 中 添加 一 个 命名 锚 记 ， 实 现 同一 页 面 的 引用 跳 转 ， 最 后 ， 
在 输出 引用 内 容 的 文本 域 中 ， 根 据 超 链接 中 传递 的 栏目 标识 ， 从 数据 库 中 读 取 到 指定 的 回帖 数据 ， 将 
引用 的 内 容 进 行 输出 。 

(1) 在 send_forum_content.php 文件 中 通过 创建 引用 的 超 链接 ， 并 且 设 置 链接 的 栏目 标识 cite 和 

锚 点 bottom 来 实现 引用 的 操作 。 关 键 代码 如 下 : 

<a href="send_forum_content.php?send_big_type=<?php echo 


$_GET["send_big_type"];?>&&send_small_type=<?php echo $_GET["send_small_type"];?>&&send_id=<?php 
echo $_GET["send id"];?>&&cite=<?php echo $myrow_4["tb_restore_id"];?>#bottom"> 引 用 </a> 


(2) 在 指定 的 位 置 设置 一 个 命名 锚 记 ， 实 现 同一 页 面 的 跳 转 。 代 码 如 下 : 


<a name="bottom" id="bottom"></a> 


(3) 在 要 输出 引用 内 容 的 文本 域 中 进行 编辑 ， 根 据 超 链接 栏目 标识 cite 的 值 ， 从 数据 库 中 读 取 到 
对 应 的 回复 帖子 的 标题 和 内 容 ， 并 且 将 读 取 的 数据 放 入 div 标签 后 输出 到 文本 域 中 ， 用 于 分 隔 引用 和 
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回复 的 内 容 。 代 码 如 下 : 


<td width="641"><input name="restore_subject" type="text" id="restore_subject" value=" 
<?php 
if($_GETI"cite"]==trueX{ 
$query=mysql_query("select * from tb_forum_send where tb_send _id='$_GETI[send_id]"); 
Sresult=mysql_fetch_array($query); 
echo "摘自 (".$result["tb_send_user”].") : ".$result["tb_send_subject"]; 
?>" size="80"> 
</td> 
<!-- 通 过 文本 框 输出 引用 内 容 --> 
<td width="641"><textarea name="file"” cols="80" rows="10" id="file” onKeyDown="countstrbyte (this.form. 
file,this. form.total,this.form.used ,this.form.remain);” onKeyUp=" countstrbyte (this.form. file,this.form.total, this. 
orm.used, this.form.remain);"> 
<?php 
if($_GETI"cite"]==trueX{ 
$query=mysql_query("select * from tb_forum_restore where tb_restore_ id='$_GETI[cite]"); 
$result=mysql_fetch_array($query); /根据 ID 从 数据 库 中 获取 到 引用 的 内 容 
// 输 出 引用 内 容 放 入 类 为 reply 的 div 内 ， 用 于 分 隔 引 用 和 回复 的 内 容 
echo "<div class=\"reply\">". $result["tb_restore_content"]."</div>"; 
} 
?> 
</textarea> 
</td> 


帖子 引用 技术 是 通过 上 述 方法 实现 的 ， 然 后 就 可 以 将 引用 的 内 容 直 接 进行 提交 ， 作 为 自己 的 回复 
帖子 。 有 关 该 技术 的 完整 应 用 可 以 参考 本 模块 中 的 send_forum_content.php 文件 。 


7.2.4 ”帖子 收藏 


帖子 收藏 就 是 将 当前 帖子 的 地 址 完整 地 保存 到 指定 位 置 ， 为 以 后 访问 该 帖子 提供 方便 。 帖 子 收藏 
的 操作 流程 如 图 7.12 和 图 7.13 所 示 。 
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标题 |PHP6 与 PHP5 有 什么 不 同 之 处 ? 


同姓 : |http://192. 163. 1. 59/NR/12/send_forun_contert. php?send_big_type=: 
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”了 编 社 英之 闲 


图 7.13 获取 帖子 收藏 地 址 


帖子 收藏 实现 的 关键 是 如 何 获取 当前 页 面 的 完整 地 址 。 获 取 当 前 页 面 的 完整 地 址 主要 是 通过 服务 


器 变量 $_ SERVER 来 获取 收藏 帖子 的 URL 地 址 来 实现 的 。 
帖子 收藏 是 在 send_forum_content.php 文件 中 实现 的 。 关 键 代码 如 下 : 
$self=$_SERVER[HTTP_REFERERI; /| 获取 链接 到 当前 页 面 的 前 一 页 面 的 url 地 址 
$u=$_SERVER[HTTP_HOST]; /获取 当前 请 求 的 Host 头 信息 的 内 容 
$r=$_SERVER[PHP_SELF'; /获取 当前 正在 执行 脚本 的 文件 名 
$I=$_SERVER[QUERY_STRING; /| 获取 查询 query) 的 字符 串 url 中 第 一 个 问号 之 后 的 内 容 ) 
$url="http:/"."$u"."$r"."?"."$I"; // 将 获取 的 变量 组 成 一 个 字符 串 ， 即 完整 的 路 径 
在 获取 到 完整 的 地 址 之 后 ， 接 下 来 将 帖子 的 标题 、 当 前 页 面 的 完整 路 径 和 当前 用 户 数 据 提交 到 


my_collection.php 页 中 ， 生 成 一 个 表单 ， 最 后 将 数据 提交 到 my_collection_ok.php 文件 中 ， 完 成 帖子 的 


收藏 。 提 


交 帖 子 完 整地 址 和 帖子 标题 的 程序 代码 如 下 : 


<?php if( lempty($_SESSION["'tb forum_user])){ ?> 
<form name="form1" method="post" action="my_collection.php?forum_subject=<?php echo 
$myrow_3["tb_send_subject"];?>&&collection_user=<?php echo $_SESSION["tb_forum_user"];?>"> 
<td width="173" height="22" align="center" valign="bottom"> 
<input type="hidden" name="my_collection" value="<?php echo $url;?>"> 
<input type="submit" name="Submit" value=" 添加 到 我 的 收藏 夹 "> 


</td> 

</form> 

<?php} ?> 

然后 创建 my_collection.php 文件 ， 生 成 一 个 表单 ， 为 收藏 的 帖子 添加 标签 和 说 明 ， 最 后 将 数据 提 


交 到 my_collection_ok.php 文件 中 ， 将 帖子 收藏 的 数据 添加 到 指定 的 数据 表 中 。 至 此 ， 帖 子 收藏 技术 讲 


解 完 毕 ， 
7.2.5 


屏蔽 


相关 的 程序 代码 请 参考 本 书 附带 光盘 中 的 内 容 ， 这 里 不 再 獒 述 。 
屏蔽 回帖 


回帖 是 管理 员 的 权限 ， 在 论坛 的 后 台 管 理 中 进行 操作 。 帖 子 是 否 被 屏蔽 是 根据 回复 帖子 数据 


表 中 tb_restore tag 字段 的 值 来 判断 的 ， 如 果 帖 子 tb_restore_tag 字段 的 值 为 1， 则 说 明 该 帖子 被 屏蔽 ; 
否则 帖子 没有 被 屏蔽 。 因 此 ， 屏 蔽 回帖 就 是 将 指定 帖子 的 tb_restore tag 字段 的 值 更 新 为 1。 


屏蔽 


回帖 主要 通过 两 个 文件 来 完成 : 一 个 是 message_restore.php， 输 出 回复 帖子 的 内 容 ， 创 建 执行 


_ 国 
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屏蔽 帖子 的 form 表单 ; 另 一 个 是 message_restore_ok.php， 根 据 提交 的 数据 ， 实 现 屏蔽 帖子 的 操作 。 这 
里 只 给 出 屏蔽 回帖 的 处 理 文件 message_restore_ok.php 的 相关 程序 ， 其 重要 代码 如 下 : 


<?php session_start(); include("conn/conn.php"); 


$Submit=$_POSTI"Submit"]; /获取 按钮 值 
if($Submit==" 屏 项 "){ // 判 断 当 提交 的 值 等 于 屏蔽 时 
whilellist($name,$value)=each($_POST)){ /获取 表单 提交 值 并 循环 输出 


$result=mysql_query("update tb _ forum_restore set tb_restore_tag="1' where tb_restore id=".$name.""); 
/根据 ID 的 值 执行 循环 操作 
if($result==trueX{ 
echo "<script>alert(' 屏 项 成 功 !); window.location.href='index.php?title= 回 帖 管理 ;</script>";} 


和 
if($Submit2==" 取 消 "){ // 判 断 当 表单 提交 的 值 等 于 “取消 ”时 
whilellist($name,$value)=each($_POST)){ // 获 取 表 单 提交 的 值 
$result=mysql_query("update tb_forum_restore set 了 b_restore tag='0' where tb_restore_id=".$name.""); 
/执行 更 新 操作 


if($result==trueX{ 
echo "<script>alert(' 取 消 屏蔽 !); window.location.href='index.php?title= 回 帖 管理 ;</script>";}} 


} 


7.2.6 ”连接 远程 MySQL 数据 库 


在 本 模块 中 使 用 的 是 MySQL 数据 库 ， 因 此 实现 PHP 与 MySQL 数据 库 的 连接 是 一 个 非常 关键 的 
步骤 ， 它 是 执行 其 他 操作 的 先决 条 件 。 在 开发 程序 的 过 程 中 ， 多 数 情况 下 使 用 的 都 是 本 地 MySQL 数 
据 库 ， 下 面 介绍 一 种 连接 远程 MySQL 数据 库 服 务 器 的 方法 。 

首先 ,在 远程 数据 库 服务 器 的 机 器 中 , 通过 MySQL 数据 库 的 图 形 化 管理 工具 phpMyAdmin， 在 权 
限 中 创建 一 个 新 用 户 ， 设 置 用 户 名 为 root， 密 码 为 111， 主 机 为 程序 运行 计算 机 的 IP 地 址 。 操 作 步 又 
如 图 7.14 一 图 7.16 所 示 。 


de 


EE 
Er 


合计 择 效 卫 库 


7.14 选择 权限 


他 
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个 全 政 / 全 部 不 选 


全 -二 新 用 户 选择 添加 新 用 户 


一 际 造 中 
TT FT 
玛 陆 与 用 户 才 季 相同 的 办 所 


多 
更 改 痘 录 信 息 /复制 用 户 . 
登入 信息 


用 户 各 [使 用 文本 


oo 


ESE 


127001 
密码 | 使 用 文本 域 


重新 输入 


Genarate Password [Generate | [Copy] 


by 
留 旧 用 户 。 


o 必用 重用 
晤 “收回 旧 用 户 的 所 有 激活 相 限 ， 然 后 删除 卓 用户。 
口 “ 从 用 户 素 中 删除 日 用 户 ， 短 后 重新 载 入 权限 ， 


EI 


局 地 地 Intzanet| 保志 本 不; 要 用 证 -可 100% 。 


图 7.16 添加 新 用 户 


只 
是 root， 密 码 是 111， 主 机 是 192.168.1.149，192.168.1.149 是 程序 运行 所 在 计算 机 的 IP 地 址 。 


新 用 户 创建 完成 之 后 连接 远程 的 MySQL 数据 库 服务 器 。 使 用 的 函数 与 连接 本 地 服务 器 相同 。 程 
序 代码 如 下 : 


<?php 
$conn=mysql_connect('192.168.1.99','root,’111"); // 连 接 数据 库 服务 器 192.168.1.99 


_- 国 
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mysql_select_db(db forum',$conn); /| 连接 db_forum 数据 库 


mysql_query("set names gb2312"); // 指 定数 据 库 中 字符 的 编码 格式 
?> 


人 MySQL 数据 库 服务 器 的 了 地址 ， 用 户 名 是 root， 密 码 是 111. 


7.2.7 “小 纸 条 信息 的 无 刷新 输出 


小 纸 条 信息 的 无 刷新 输出 主要 应 用 的 是 Ajax 技术 。 通 过 Ajax 技术 调用 指定 的 文件 查询 是 否 存在 
新 的 消息 ， 并 且 将 结果 返回 ， 通 过 div 输出 Ajax 中 返回 的 查询 结果 。 运 行 结果 如 图 7.17 所 示 。 


7.17 小 纸 条 信息 无 刷新 输出 


关键 的 Ajax 代码 如 下 : 


<script type="text/javascript" src="js/xmlHttpRequest.js"></script> 
<script language="javascript"> 


function show_counts(sender){ 1// 通 过 show_counts() 函 数 调用 指定 的 文件 统计 最 新 信息 
url='show_counts.php?sender='+sender; /| 设置 要 执行 的 文件 ， 并 设置 栏目 标识 
xmlHttp.open("get",url, true); /执行 操作 


xmlHttp.onreadystatechange = function(X{ 
if(xmlHttp.readyState == 4X{ 
tet = xmlHttp.responseText; 
show_counts11.innerhtml=tet;”// 返 回 统 计 结果 
} 
} 
xmlHttp.send(null); 
</script> 
<!-- 间 隔 1000 毫秒 执行 一 次 show_counts() 函 数 -> 
<script language="javascript"> 
setinterval("show_counts('<?php echo $_SESSION["tb_forum_user"];?>")",1000); 
</script> 
<!-- 输 出 最 新 消息 使 用 的 是 div 标签 -> 
<td width="15" align="center" class="STYLE2"><div id="show_counts11"></div></td> 


7.2.8 清除 个 人 站 内 邮件 
一 个 完善 的 论坛 系统 离 不 开 站 内 邮件 的 应 用 ， 会 员 在 浏览 信件 后 ， 可 能 产生 大 量 过 期 邮件 ， 这 些 


过 期 数据 既 给 用 户 在 邮件 查找 上 带 来 麻烦 ， 又 给 数据 库 增添 了 额外 负担 。 这 时 用 户 可 以 通过 筛选 ， 对 
站 内 邮件 有 选择 地 删除 。 站 内 邮件 的 删除 过 程 如 图 7.18 所 示 。 
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上 eonsk 你 好: 个 现 在 有 0 条 未 读 信息 + 
收 件 箱 发 件 箱 
操作 标题 
口 你 好 ，leonsk 
[| i 你 好 ，leonsk 
所 一 | 一 划 方 已 经 通过 好 友 申请 验证 
加 * 对 方 已 经 通过 好 友 申 请 验证 
[| 对 方 已 经 通过 好 友 申 请 验证 
页 次 :1 / 2 页 记录 :13 条 


清 队 信息 写 术 
发 御 人 发 送 时 间 清除 信息 
adnin 2011-01-12 已 读 信息 
adnin 2011-01-12 已 读 信息 
adnin 2011-01-13 已 读 信息 
adnin 2011-01-13 已 读 信息 
系统 信息 2011-01-13 已 读 信息 


分 页 : 下 一 页 尾 页 


图 7.18 站 内 邮件 的 删除 过 程 


获取 要 删除 的 站 内 邮件 id 后 ， 将 选 定 的 id 值 提交 到 delete_mail.php 文件 中 ， 完 成 个 人 站 内 邮件 删 
除 操作 。 选 择 删除 站 内 邮件 的 文件 browse_mail.php 的 关键 代码 如 下 : 


<form name="form1" method="post" action="delete_mail.php?sender=<?php echo $_GET["sender"];?>&&mails= 


<?php echo $_GET["mails"];?>"> 
<tr> 


<td width="77" height="35" align="center"><strong><span class="STYLE_mail"> 操 作 </span></strong></td> 


<td width="266" align="center"><strong><span class="STYLE_mail"> 标 题 </span></strong></td> 


<td width="178" align="center"><strong><span class="STYLE_mail"> 发 件 人 </span></strong></td> 

<td width="206" align="center"><strong><span class="STYLE_mail"> 发 送 时 间 </span></strong></td> 

<td width="189"”height="35”align="center”class="STYLE_mai"><input name="Submit” type="submit" 
id="Submit" value=” 清 除 信 息 "></td> 


</tr> 
<?php 
if($pageX{ 
$page_size=10; 
// 从 数据 库 中 读 取 数 据 


$query="select count(*) as total from tb_mail_box where tb_receiving_person='$_GET[Isenden' "; 


$result=mysql_query($query); 


$message_count=mysql_result($result,0,"total"); 

$page_count=ceil($message_count/$page_size); 

S$offset=($page-1)*$page_size; 
$query=mysql_query("select * from tb_mail_box where tb_receiving_person='$_GET[sender]' order by tb_mail_ 
type="1' limit $offset, $page_size"); 
while($myrow=mysql_fetch_array($query)\{ 


?> 
<tr> 


// 每 页 显示 10 条 记录 


// 获 取 总 的 记录 数 


/获取 总 的 页 数 


<td height="35" align="center><input name="<?php echo $myrow["tb_mail_id"];?>" type="checkbox" value= 
"<?php echo $myrow["tb_mail id"];?>"> </td> 
<td align="center"><a href="browse_mail_content.php?mail_id=<?php echo $myrow["tb_mail_id"];?>" target= 
"_blank" class="STYLE3_mail"><?php echo $myrow["tb_mail_subject"];?></a></td> 


入 
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<td align="center"><span class="STYLE3_mail"><?php echo $myrow["tb_mail_sender"];?></span></td> 
<td align="center"><span class="STYLE3_mail"><?php echo $myrow["tb_mail_date"];?></span></td> 
<td align="center"><span class="STYLE3_mail"> 
<?php if($myrow["tb_mail_type"]=='1'){echo "已 读 信息 ";}else{echo "未 读 信息 ";}?> 
</span></td> 
</tr> 
<?php }?> 
</form> 


在 获取 要 删除 邮件 的 id 后 ， 将 id 提交 到 delete_mail.php 页 中 ， 完 成 选中 邮件 的 删除 操作 。 删 除 站 
内 邮件 的 程序 代码 如 下 : 


<?php session_start(); include("conn/conn.php"); 
$Submit=$_POST[ Submit]; // 获 取 提 交 的 Submit 值 
$mails=$_GET[mails]; 
if($Submit==" 清除 信息 "{ 
whilellist($name,$value)=each($_POST)){ 


$array[]=$name; // 将 获得 的 id 元 素 添加 到 数组 中 
和 
if(count($array)-1>0X // 如 果 获 得 的 总 数 减 1 大 于 0， 则 说 明 有 选中 的 删除 元 素 


for($i=1;$i<count($array);$i++){ ””// 循 环 删除 选中 的 信息 
$result=mysql_query("delete from tb_mail_box where tb_mail_id=".$array[$i].""); 


} 
echo "<script>alert( 删 除 成 功 !); window.location.href='send_mail.php?sender=$_GET[sender] 
&&mails=$mails';</script>"; 
}else{ 
echo "<script>alert(' 请 先 选择 要 删除 的 信件 !);history.back();</script>";， /如 果 没 有 选择 删除 的 
文件 ， 则 给 出 一 个 提示 


?> 


首先 获取 从 browse_mail.php 中 得 到 的 Submit 传递 值 和 选中 邮件 的 id 值 ,如 果 该 页 面 获 得 的 Submit 
传递 值 为 “清除 信息 ”， 则 继续 判断 选中 邮件 id 的 数量 。 


~, 
后 曙 利用 POST 方式 传递 相关 参数 ， 其 中 前 台 表 单 域 的 Submit 也 是 利用 该 方式 进行 传递 ， 所 
以 Submit 也 会 在 后 台 处 理 页 作为 一 个 元 素 被 添加 到 数组 中 ， 在 判断 选 定 id 值 的 个 数 时 ， 需 要 执行 
减 1 操作 ， 在 执行 循环 操作 的 过 程 中 也 需要 跳 过 此 数组 中 的 值 ， 以 免得 到 不 必要 的 结果 。 


然后 ， 判 断 选择 邮件 id 的 数量 ， 如 果 选 择 邮件 id 的 数量 为 0， 则 给 出 当前 会 员 警告 提示 ， 告 之 其 
不 能 进行 删除 操作 ;如 果 判 断 选择 的 邮件 id 数量 不 为 0， 则 继续 向 下 执行 其 他 操作 。 
最 后 ， 通 过 选择 的 id 来 删除 数据 库 中 的 邮件 内 容 ， 从 而 实现 邮件 删除 操作 。 
至 此 ， 站 内 邮件 清除 技术 讲解 完毕 ， 此 技术 同样 应 用 于 好 友 删 除 技术 中 。 
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7.3 ”数据库 设计 


7.3.1 数据 库 分 析 


论坛 的 功能 完善 与 否 ， 数 据 库 的 运用 是 一 个 决定 性 的 因素 。 只 有 拥有 一 个 强大 数据 库 的 支持 ， 论 
坛 的 功能 才能 够 展现 ， 否 则 它 将 和 留言 簿 没什么 区 别 。 

本 论坛 中 使 用 的 是 一 个 名 称 为 db_forum 的 数据 库 ， 在 该 数据 库 中 有 9 个 数据 表 。 有 关 数 据 表 名 称 
及 表 功 能 的 介绍 如 图 7.19 所 示 。 
i 如 SQL 户 扫 过 | 喇 查 询 。 革 导 出 加 Import 24 操作 s 吉 权限 E 要 ps 

胡 类 型 整理 这 明 

th_forum_amiche MyISAM gb2312_chinese_ci ”存储 公告 信息 的 数据 表 
tb_Torum_plg_type MylSAM gb2312_chinese_cl 存储 所 原 专 区 BI 关 别 | 
th_forum_restore MyISAM gb2312_chinese_ci ”存储 论坛 回复 的 其 据 
th_forum_sond MylsAM gb2312_chinese_cl 存储 论坛 发 送 的 数据 
forum_small_type MySAM ”gb2312_chinese_cl ， 存 诗 论 坛 中 帖子 的 闪 别 
tb_Torum_user MyISAM ”gb2312_chinese_cl ， 存 针 用 户 的 注册 信息 
th_mall_box MylISAM gb2312_chinese_cl ”存储 短信 信息 


th_my_collection MyISAM ”gb2312_chinese_ci 。 行 竺 我 收 牙 的 妆 据 
th_my_mmend MSAM gp2312_chinese_cl ”存储 息 B0 好 友信 息 


图 7.19 db_forum 数据 库 中 的 数据 表 


7.3.2 创建 数据 库 中 的 数据 表 


下 面 对 数 据 库 中 儿 个 相对 比较 复杂 的 数据 表 的 功能 和 结构 进行 介绍 。 

tb_forum_user 数据 表 : 用 于 存储 用 户 的 注册 信息 。 其 中 包括 13 个 字段 , 字段 属性 的 说 明 如 图 7.20 
所 示 。 

tb_forum_send 数据 表 : 用 于 存储 论坛 中 发 布 帖子 的 数据 。 其 中 包括 11 个 字段 ， 字 段 属性 
说 明 如 图 7.21 所 示 。 


localhost 概 季 :db_forum 


过 


四 服务 器 : loealhost > 局 数据 库 : db_forum ， 国 表 :也 _forum_user 
辕 训 蜂 _ 啊 续 构 ， ,只 SoL 户 扫 索 3 新 入 加 号 出 _ 隧 Import 膏 典 作 再 洁 空 “四则 除 

字 恬 类 型 可 得 说 明 | 加 服务 器 : localhost ， 竖 歼 据 库 : db_forum 》 国 表 : tb_forum_send 
也 forum 这 intdo) 访 轩 到 居家 朋 关 键 闻 EC 箭 叶 出 联 Imporl， 估 坎 作 而 洁 空 器 田 除 | 
th_forum_user arthan(50] gb2312_chinese_ci 创建 的 用 户 名 FB 委 型 要 型 说 明 
人 b_forum_pass Varthar(50] gb2312_chinese_ci 。 注册 的 室 码 nk10) 设 轩 归 据 表 的 主 时 
hforum pe varchar(50] 9b2312_chinese_i 。 用 户 得 奖 型 ，; 力 会 员 ; 2 力 交 生 民 send_subject archarG0) gb2312_thinese_cl 发 于 的 主 是 
th_forum_email Warthalt50) 0b2312_chinese_ci 。 注册 9 二 箱 [PT meciurrted gb2312_ehinese_ri 发布 帖 子 的 内 容 
tb_forum_truepass Varthats0j 9b2312_chinese_cl 。 注入 纹 真 实 至 码 I 0 
‘thforum._ dete atme Emela bp seng_date ahme 邮 子 后 时 间 
th forum picture vartharf80] 0b2312_chinese_o 。 尖 hrA9 丛 ‘th_send_picture atehar00) gb2312_thinese_di 素 情 加 
Th_forum_grade Varehat(50] gb2312_chinese_cl 。 会 员 99 积 分 了 b_send_type VarcharG0) gb2312_chinsse_ci 着 册 | 帖子 是 否 蛋 项 ,1 为 置顶 
也 _forum_pass_problom varchat(50] gb2312_chinese_ci 。 客 友 提示 问题 WD_send ypes “atnar(50) gb2312 China$e_Cl 钛 采 帖 子 是 否 已 经 回复 ，1 为 回复 
也 _forum_pass resukt 。 Yarthalf50) gb2312_chinese_ci 窗 码 提示 的 等 宗 bh_send_small_type archarG0) gb2312_chinase_ci 帖子 所 属 的 关 别 
hforum_qq Varharsol gb2312_chinese_d 。 王 由 HQ send_ype_distllate varcner(S0) gb2312_chinese_di 和 断 由 于 是 玉 半 第 ,1 为 村 由 
th_forum specialty ted 9b2312_ehinese_di 。 汗 的 个 从 攻 四 send_Wpe_halspot acnant50) ob2312_thinese_C| 所 断 由 子 是 开 为 站 由。 1 为 扫 门 相 

图 7.20 ”tb_forum_user 数据 表 7.21 tb_forum_send 数据 表 
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回 tb_forum_ restore 数据 表 : 用 于 存储 论坛 中 回复 帖子 的 数据 。 其 中 包括 7 个 字段 ， 字 段 属性 的 
说 明 如 图 7.22 所 示 。 

罚 服务 器 : localhost ”总 数据 库 : db_forum 》 国 表 :tb_forum_restore 

|_- 国 训 客 “ 国 伟 构 “ 玉 Sol 万 殷 索 也 :英和 圈 导 出 并 Import 伏 换 作 [ 别 清 空 央 则 除 
字 良 类 型 整理 说 明 

也 restore dl int1g) 设置 数 括 表 的 主 当 

tb_restore_subject varchar(50) ”gb2312_chinese_cl 回复 帖子 的 主题 

tb_restore_content mediumtext gb2312_chinese_ci 回复 帖子 的 内 容 

th_restore_user 。 varchar(50) gb2312_chinese_ci 回复 者 

th_send_id Int1g) 发 帖子 的 TD 

th_restore date datetime 回复 时 间 

OY | 下 La、 -| 


图 7.22 tb_forum_restore 数据 表 

tb_my_collection 数据 表 : 用 于 存储 用 户 收藏 的 帖子 。 其 中 包括 7 个 字段 ， 字 段 属 性 的 说 明 如 
图 7.23 所 示 。 

困 服务 器 : localhost ， 电 数据 库 : db_forum ， 国 表 : tb_my_collection 


训 览 _ 阿 结 构 ，, 系 SOL 户 扫 索 和 新 入 障 导 出 阐 Import 允 换 作 ， 曾 清空 ， 况 开除 
字 及 类 型 整理 说 明 


也 _collection id int(10) 设置 数据 表 的 ID 
也 _collection_subject 。 varchar(50) 。 gb2312_chinese_ci 收 若 帖子 的 标题 
也 _collection_address 。 varchar(150) gb2312_chinese_ci 收 茂 帖子 的 地 址 
th_collection_label varchar(50) 。 gb2312_chinese_ci 收 鞍 帖子 的 标签 
也 _collection_summary mediumtex 。 gb2312_chinese_ci 收 若 帖子 的 说明 
th_collection_user varchar(50) ”gb2312_chinese_ci 收藏 者 

th_collection_date datetime 收 套 时 间 


图 7.23 tb_my_collection 数据 表 


在 本 模块 中 包括 9 个 数据 表 ， 由 于 篇 幅 所 限 ， 这 里 只 介绍 了 其 中 4 个 相对 比较 复杂 的 数据 表 ， 有 
关 其 他 数据 表 的 内 容 和 属性 可 以 参考 本 书 附带 光盘 。 


7.4 帖子 的 发 布 、 浏 览 和 回复 


7.4.1 帖子 的 发 布 、 浏 览 和 回复 概述 


本 论坛 模块 帖子 功能 包括 帖子 的 发 布 、 帖 子 的 浏览 以 及 帖子 的 回复 等 。 其 中 ， 帖 子 发 布 为 登录 的 
会 员 提供 一 个 发 布 帖子 的 操作 平台 。 在 该 平台 中 ， 登 录用 户 可 以 选择 发 布 帖子 的 类 别 、 自 定义 帖子 的 
主题 、 选 择 表情 图 、 选 择 上 传 附件 以 及 通过 文本 编辑 器 对 发 布 帖子 的 内 容 进行 编辑 。 帖 子 发 布 的 运行 
结果 如 图 7.24 所 示 。 

帖子 回复 实现 对 指定 的 帖子 进行 回复 的 操作 ， 帖 子 回复 的 运行 结果 如 图 7.25 所 示 。 

帖子 浏览 包括 帖子 主题 和 帖子 内 容 的 浏览 。 首 先 可 以 浏览 到 根据 不 同类 别 进行 划分 的 帖子 主题 ， 
然后 可 以 在 相应 的 帖子 主题 中 浏览 到 具体 帖子 的 内 容 。 帖 子 主题 和 帖子 内 容 浏 览 的 运行 结果 如 图 7.26 
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和 图 7.27 所 示 。 
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图 7.24 帖子 发 布 的 运行 结果 
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图 7.26 帖子 主题 浏览 的 运行 结果 
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7.27 帖子 内 容 浏览 的 运行 结果 
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7.4.2 ”帖子 发 布 功能 实现 


帖子 发 布 主要 由 两 个 文件 构成 : 一 个 是 帖子 发 布 内 容 的 填写 文件 send_forum.php; 另 一 个 是 提交 
数据 的 处 理 文件 send_forum_ok.php。 

在 send_forum.php 文件 中 ， 可 以 将 该 文件 中 的 内 容 分 成 3 个 部 分 : 第 1 部 分 初始 化 Session 变量 ， 
连接 数据 库 以 及 调用 js 文件 ; 第 2 部 分 输出 当前 登录 会 员 的 个 人 信息 ; 第 3 部 分 构建 fom 表单 ， 实 现 
发 布 帖子 数据 的 提交 。 

(1) 初始 化 Session 变量 ， 连 接 数据 库 以 及 调用 指定 的 包含 文件 ， 并 且 判 断 当前 用 户 是 否 是 会 员 ， 
如 果 不 是 会 员 将 不 能 进行 帖子 发 布 的 操作 。 

帖子 的 发 布 文件 send_forum.php 的 关键 代码 如 下 : 

<?php session_start(); include("conn/conn.php"); 

if($_SESSION["tb_forum_user"]==trueX{ // 判 断 当 以 会 员 登 录 时 执行 下 面 的 内 容 

Sr 
< type="text/javascript" src="js/editor.js"></script> 


(2) 从 数据 库 中 读 取出 当前 会 员 的 个 人 信息 ， 并 且 将 个 人 信息 输出 。 代 码 如 下 : 


<?php $query_1=mysql_query("select * from tb_forum_user where tb_forum_user='$_SESSION["tb forum_ 
user]",$conn); 

$myrow_1=mysql_fetch_array($query_1);// 执 行 查询 语句 ， 获 取 当 前 登录 用 户 的 个 人 信息 
echo "<img src='$myrow_1["tb_ forum_picture"]>"; 

echo "<br>"; 

echo "当前 用 户 :"; 

echo "<br>"; 

echo $myrow_1 ["tb_forum_user"]; 

echo "<br>"; 

echo "注册 时 间 :"; 

echo "<br>"; 

echo $myrow_1["tb_forum_date"]; 

echo "<br>"; 

echo "积分 :"; 

echo $myrow_1["tb_forum_grade"]; 

?> 


(3) 创建 form 表单 ， 提 交 发 布 帖子 的 数据 ， 包 括 帖 子 的 类 别 、 主 题 、 表 情 图 、 文 本 内 容 和 发 布 
人 。 在 对 帖子 内 容 进 行 填写 时 ， 应 用 的 是 一 个 文本 编辑 器 ， 通 过 文本 编辑 器 可 以 对 提交 的 内 容 进 行 编 
辑 。 程 序 关键 代 码 如 下 : 
<!- 将 帖子 类 别 输出 到 下 拉 列 表 框 中 -> 
<td align="center"> 帖 子 类 别 :</td> 
<td><select name="send_sort" id="send_sort"> 
<option selected="selected"> 请 选择 帖子 的 类 别 </option> 
<?php /输出 帖子 的 类 别 


$query _ 2=mysql_query("select * from tb_forum_small_type"); 
@ 
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while($myrow_2=mysql_fetch_array($query_2)X 
i 
<option value="<?php echo $myrow_2["tb_small_ type_content"];?>"> 
<?php echo $myrow_2["tb_small_type_content"];?> 
</option> 
<?php }?> 
</select> 
</td> 
<!-- 表 情 图 的 排列 输出 -> 
<td align="right" class="STYLE11"> 表 情 图 : </td> 
<td><table> 
<tr> 
<td height="80" colspan="2"><div align="center"> 
<table height="30" border="0" align="center cellpadding="0" cellspacing="0"> 


<tr> 
<?php 
for($i=1;$i<=24;$i++){ /根据 文件 来 中 表情 图 的 个 数 创建 循环 语句 
if($i%6==0){ // 判 断 变量 的 值 是 否 等 于 0 
?> 


<td width="40" height="30"><div align="center"><!-- 输 出 表情 图 --> 
<img src=<?php echo("images/inchoative/face".($i-1).".gif’);?> width="20" height= 
"20"></div></td> 
<td width="40" height="30"><div align="center"><!-- 创 建 单 选 按钮 -> 
<input type="radio" name="face" value="<?php echo " ("images/ inchoative/ 
face.($i-1).".gif’);?>"> 
</div></td> 
</tr> 
<?php }else{?> 
<td width="40" height="30"><div align="center"> 
<img src=<?php echo("images/inchoative/face".($i-1).".gif");?> width="20”height= 
"20"></div></td> 
<td width="40" height="30"><div align="center"> 
<input type="radio” name="face” value="<?php echo("images/ inchoative/face". 
($i-1).".gif’);?>" <?php if($i==1) { echo "checked";}?>> 
</div></td> 
<?php } }?> 
</table> 
</div></td> 
</tr> 
</table></td> 
<!-- 通 过 文本 编辑 器 ， 对 帖子 的 内 容 进行 编辑 -> 
<tr> 
<td width="107" align="right" class="STYLE11"> 文 章 内 容 : </td> 
<td width="569"> 
<textarea name="menu" cols="1" rows="1" id="menu" style="position:absolute;left:0;visibility:hidden;"></textarea> 
<script type="text/javascript"> 
var editor = new FtEditor("editor"); 
editor.hiddenName = "menu"; 
editor.editorWidth = "100%"; 
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editor.editorHeight = "300px"; 

editor.show(); 

</script> 

<input type="hidden" name="tb_forum_user" value="<?php echo $_SESSION["tb_forum_user"];?>"></td> 
</tr> 


有 关 send_forum.php 文件 的 讲解 到 此 结束 ， 完 整 代码 请 参考 本 书 附带 光盘 中 的 内 容 。 
下 面 介绍 表单 处 理 页 send_forum_ok.php 文件 。 在 该 文件 中 将 表单 中 提交 的 数据 存储 到 数据 库 中 ， 


发 布 帖子 信息 的 存储 。 帖 子 发 布 处 理 页 send_forum_ok.php 文件 的 关键 代码 如 下 : 
<?php session_start(); include_once("conn/conn.php"); 

S$tb_send_type=0; // 设 置 帖子 是 否 置 项 
$tb_send_types=0; /| 判断 帖 子 是 否 有 回复 
S$tb_send_small_type=$_POST["send_sort"]; // 获 取 表 单 中 提交 的 数据 
S$tb_send_subject=$_POSTI["send_subject"]; /| 获取 表单 中 提交 的 数据 
$tb_send_picture=$_POST["face"]; /获取 表单 中 提交 的 数据 
$tb_send_content=trim($_POST["menu"]); // 获 取 表 单 中 提交 的 数据 


$tb_send_user=$_POST['tb forum_user]; 
$tb_send_date=date("Y-m-j Hii:s"); 
if($_FILES[send_accessoriesj[size]==0)f 1/ 判断 是 否 有 附件 上 传 
// 执 行 添加 语句 
这 mysqlL_query("insert into tb_forum_send(tb_send_subject,tb_send_content,tb_send_user,tb_send_date,tb_ 
send_picture,tb_send_type,tb_send_types,tb_send_small_type,tb_send_authortb_send_ltime) values (".$tb_ 
send_subject.",".$tb_send_content.",".$tb_send_user.",".$tb_send_date.",".$tb_send_picture.",".$tb_send_ 
type.",".$tb_send_types.",".$tb_send_small_type.",".$tb_send_user.",".$tb_send_date.")",$conn))f 
mysql_query("update tb_forum_user set tb_forum_grade=tb_forum_grade+5",$conn); 
echo "<script>alert( 新 帖 发 表 成 功 !);history.back();</script>"; 
mysql_close($conn); 
Jelse{ 
echo "<script>alert(' 新 帖 发 表 失败 !");history.back();</script>"; 
mysql_close($conn); 


» 
} 
if($_FILES["send_accessories][size"] > 20000000X{ /判断 上 传 附 件 是 否 超过 指定 的 大 小 
echo "<script>alert(' 上 传 文件 超过 指定 大 小 ! ');history.go(-1);</script>"; 
exit(); 
}else{ 
$path = '/file/.time().$_FILES[send_accessories]rname]; /定义 上 传 文件 的 路 径 和 名 称 


if (move_uploaded file($_FILES[send_accessories][tmp_name],$path)){ /将 附件 存储 到 指定 位 置 
if(mysql_query("insert into tb_forum_send(tb_send_subject,tb_send_content,tb_send_user,tb_send_date, 
tb_send_picture,tb_send_type,tb_send _ types,tb_send_small_type,tb_send_accessories,tb_send_authortb_se 
nd ltime) values 〈".$tb_send_subject",".$tb_send_content",".$tb_send_user",".$tb_send_date.",".$tb_send _ 
picture.",".$tb_send_type.",".$tb_send_types.",".$tb_send_small_type.",".$path.",".$tb_send_user",".$tb_s 
end_date.")",$conn)){ 
// 执 行 更 新 操作 ， 为 会 员 添加 积分 
mysql_query("update tb_forum_user set tb_forum_grade=tb_forum_grade+5",$conn); 
echo "<script>alert(' 新 帖 发 表 成 功 !");history.back();</script>"; 
mysql_close($conn); 
Jelse{ 
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echo "<script>alert( 新 帖 发 表 失败 !");history.back();</script>"; 
mysql_close($conn); 


» 
?> 


其 中 , 需要 重点 说 明 的 是 , 发 新 帖 的 同时 , 会 向 最 后 回复 者 tb_send_author 和 最 后 回复 时 间 tb_send_ 
ltime 两 个 字段 中 加 入 发 帖 者 的 名 称 和 发 帖 时 间 。 
至 此 ， 帖 子 发 布 功 能 的 实现 过 程 介绍 完毕 ， 完 整 代码 可 以 参考 本 书 附带 光盘 中 的 内 容 。 


7.4.3 ”帖子 浏览 功能 实现 
帖子 浏览 是 从 帖子 类 别 的 输出 开始 的 ， 首 先 在 网 站 的 左 侧 框架 中 应 用 树 状 导 航 菜单 输出 帖子 的 类 


别 ， 根 据 树 状 导航 菜单 中 输出 帖子 的 类 别 ， 设 置 超 链 接 ， 将 指定 类 别 的 帖子 在 右 侧 的 框架 中 输出 ， 即 
在 content.php 文件 中 输出 帖子 的 内 容 。 其 中 left.php 中 的 相关 程序 关键 代码 如 下 : 


<td width="84%" height="24" onClick="javascript:open_close(id_a<?php echo $myrow["tb_big_type_id"];?>);" > 
<!-- 设 置 超 链接 ， 并 指定 target 的 值 -> 
<a href="content.php?content=<?php echo $myrow["tb_big_type_content"];?>&&content_1=<?php echo 
$myrows ['tb_small_type_content"];?>" target="contentFrame"><?php echo $myrow["'tb_big_type_content"];?></a> 
</td> 


tg 二 1 content.php 文件 是 在 右 侧 的 框架 中 输出 的 内 容 。 超 链接 中 的 target 属性 获取 的 是 
右 侧 框架 中 的 链接 文件 的 名 称 。 设置 栏目 标识 变量 content, 代表 帖子 的 所 属 专区 ，content_1 代表 帖 
子 的 类 别 。 


<td height="23"> 

<a href="content.php?content=<?php echo S$myrow["tb_big_type_content"];?>&&content_1=<?php echo 
$myrow_ 1["tb_small_type_content"];?>" target="contentFrame"><?php echo $myrow_1["tb_small type_ 
content"];?></a> 
</td> 


\ 
中 的 内 容 ， 这 里 不 再 复述 。 


然后 ， 在 contentphp 文件 中 输出 对 应 类 别 帖 子 的 内 容 。 其 中 应 用 switch 语句 ， 根 据 获取 的 栏目 标 
识 变量 $class 的 不 同 值 , 分 别 调用 不 同 的 文件 , 输出 不 同类 别 中 帖子 的 内 容 。content.php 文件 的 关键 代 
人 码 如 下 : 
(1) 判断 论坛 的 所 属 专区 和 类 别 是 否 为 室 ， 如 果 为 空 则 输出 默认 的 内 容 ， 否 则 将 输出 对 应 专区 和 
类 别 中 帖子 的 内 容 。 
<!-- 判 断 当 论坛 的 所 属 专区 和 类 别 为 空 时 ， 执 行 下 面 的 内 容 -> 


<?php if($_GET["content"]=="" and $_GET["content_1"]=="”"){ ?> 
<table><tr><td height="10">&nbsp;</td></tr> 


网 
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<tr> 
<td><?php include_once("bccd.php");?></td> <!L- 执 行 包含 文 件 -> 
</tr> 
</table> 
<?php }else{?> 


(2) 创建 一 个 搜索 引擎 的 表单 ,为 不 同 的 类 别 和 专区 设置 超 链接 , 并 设置 超 链接 的 栏目 标识 变量 。 


关键 代码 如 下 : 


<form name="form1” method="post”action="search.php?class= 搜 索引 擎 &&content=<?php echo $_GET 


[content];?>&&content_1=<?php echo $_GET[content_1];?>" onSubmit="retum check_submit();"> 
<tr><td width="10%" height="40" rowspan="2" valign="middle"> 
<a href="content.php?class= 最 新 帖子 &&content=<?php echo $_GET[content];?>&&content_1=<?php 
echo $_GET[content_1];?>"><img src="images/index_7 (1).jpg" width="65" height="23" border="0"></a></td> 
<td width="10%" rowspan="2" valign="middle"> 
<a href="content.php?class= 精 华 区 &&content=<?php echo $_GET[content];?>&&content_1=<?php 
echo $_GET[content_1];?>"><img src="images/index_7 (2).jpg" width="55" height="23" border="0"></a></td> 
<td width="10%" rowspan="2" valign="middle"> 
<a href="content.php?class= 热 点 区 &&content=<?php echo $_GET[content];?>&&content_1=<?php 
echo $_GET[content_1];?>"><img src="images/index_7 (3).jpg" width="52" height="23" border="0"></a></td> 
<td width="10%" rowspan="2" valign="middle"> 
<a href="content.php?class= 待 回复 &&content=<?php echo $_GET[content];?>&&content_1=<?php 
echo $_GET[content_1];?>"><img src="images/index_7 (4).jpg" width="55" height="23" border="0"></a></td> 
<td width="25%" height="38" align="right" valign="bottom"> 
<input name="tb_send_subject_content" type="text" size="20" />&nbsp;&nbsp;</td> 
<td width="25%" rowspan="2"> 
<input type="image" name="imageField" src="images/index_71.jpg" /></td> 


</tr> 
</form> 
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(3) 编写 switch 语句 ， 根 据 栏目 标识 变量 $class 的 不 同 值 ， 调 用 不 同 的 文件 。 代 码 如 下 : 


<?php 
if(empty($_GET["class"]) 
$class=""; // 如 果 class 没有 传递 值 ， 则 赋值 为 空 
else 
$class=$_GET["class"]; // 如 果 class 有 传递 值 ， 则 将 获得 的 class 值 赋 给 $class 
Switch($class){ 
case "最 新 帖子 ": 
include("new_forum.php"); 
break; 
case "精华 区 ": 
include("distillate.php"); 
break; 
case "热点 区 ": 
include("hotspot.php"); 
break; 
case " 待 回复 ": 
include("pending.php"); 
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include("new_forum.php"); 
break; 
} 


} 
?> 


根据 $class 变量 的 不 同 值 调用 不 同 的 文件 , 在 被 调用 的 这 些 文件 中 , 读 取 数 据 库 中 数据 的 方法 都 是 
相同 的 ， 都 是 根据 所 属 的 类 别 从 数据 库 中 读 取 出 符合 条 件 的 数据 ， 进 行 分 页 显示 。 

这 里 以 最 新 帖子 中 调用 的 new_forum.php 文件 为 例 ， 对 被 调用 文件 的 创建 方法 进行 讲解 。 

在 new_forum.php 文件 中 ， 主 要 就 是 以 超 链接 栏目 标识 中 传递 的 变量 $_GET[z"content"] 和 $_GET 
["content_1"] 为 条 件 ， 从 数据 库 中 读 取 出 符合 条 件 的 数据 。 

(1) 输出 的 是 所 属 专区 中 公告 和 置顶 帖子 的 标题 信息 ， 并 且 设 置 超 链接 ， 链 接 到 send_affiche.php 

和 send_forum_content.php 文件 ,在 对 应 的 文件 中 输出 公告 和 置顶 帖子 的 详细 内 容 。 其 中 ,new_forum.php 
文件 的 关键 代码 如 下 : 


<?php 
$query=mysql_query("select * from tb_forum_affiche where tb_affiche_type='$_GETI[content] "); 
while($myrow=mysql_fetch_array($query)X{ /| 输出 公告 内 容 
ys 
<tr> 


<td width="90%" colspan="4"><a href="send_affiche.php?tb_affiche_type=<?php echo $myrow [tb_affiche_ 
type];?>&&tb_affiche_id=<?php echo $myrow["tb_affiche_id"];?>" target="_blank"><?php echo $myrow[tb_affiche_ 
subject];?></a></td> 
</tr> 
<?php} 
$query_1=mysql_query("select * from tb_forum_send where tb_send_type="1' and tb_send_small _type=".$_ 
GET[content_1].""); 
while($myrow_1=mysql_fetch_array($query_1)X{ /| 输出 置顶 内 容 
?> 
<tr> 
<td colspan="4" width="90%"><a href="send_forum_content.php?send_big type=<?php echo $_GET 
["content"];?>&&send_small_ type=<?php echo $myrow_1["tb_send_small_ type"];?>&&send id=<?php echo 
$myrow_1 ["tb_send_id"];?>" target="_blank"><?php echo $myrow_1["tb_send_subject"];?></a></td> 
</tr> 
<?php }?> 


(2) 根据 栏目 标识 传递 的 变量 ， 从 数据 库 中 读 取出 对 应 的 专区 和 类 别 中 帖子 的 数据 ， 将 读 出 的 帖 
子 按照 最 后 发 布 的 时 间 进 行 倒序 排列 ， 这 样 即 可 实现 最 后 发 布 的 帖子 可 以 在 会 员 浏览 相关 类 别 文章 时 优 
先 被 读 取 , 之 后 定义 变量 , 实现 数据 的 分 页 显示 ; 在 输出 帖子 的 标题 时 , 设置 超 链接 , 链接 到 send_forum_ 
content.php， 在 该 文件 中 输出 帖子 的 详细 信息 。 程 序 关 键 代码 如 下 : 

<?php 


if($pageX{ 
$page_size=1; // 定 义 每 页 显示 的 条 数 


全 
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$query="select count(*) as total from tb_forum_send where tb_send_small_ type='®$_GET[content_1]"; 
$result=mysql_query($query); /执行 查询 语句 
$message_count=mysql_result($result,0,"total"); 。 ”// 获 取 总 的 记录 数 
$page_count=ceil($message_count/$page_size); 。 // 计 算 总 的 页 数 
S$offset=($page-1)*$page_size; /| 计算 上 一 页 的 最 后 一 条 记录 

/从 数据 库 中 读 取 帖子 的 数据 ， 按 照 帖 子 发 布 的 时 间 进 行 降 震 排列 输出 


$query 
tb_send 


2=mysql_query("select * from tb_forum_send where tb_send_small_type='$_GET[content_1] order by 
|_ ltime desc limit $offset, $page_size"); 


while($myrow_2=mysql_fetch_array($query_2))t/ 循 环 输出 查询 结果 


<tr> 
<td 


align="center"><img src="<?php echo $myrow_2["tb_send_picture"];?>"></td> 


<td><a href="send_forum_content.php?send_big_type=<?php echo $_GET["content"];?>&&send_small_ 
type=<?php echo $myrow_2["tb_send_small_ type"];?>&&send id=<?php echo $myrow_2["tb_send_id"];?>" 
target="_blank"><?php echo $myrow_2["tb_send_subject"];?></a></td> 

<td><?php echo $myrow_2["tb_send_date"];?></td> 

<td><?php echo $myrow_2["tb_send_user"]."/";?> 

<?php $query_ta=mysql_query("select * from tb_forum_send where tb_send_id=$myrow_2[tb_send_id]"); 

$myrow_ta=mysql_fetch_array($query_ta); 

if($myrow_ta["tb_send_types"]==1) /| 输出 最 后 回复 帖子 的 用 户 


echo $myrow _ta["tb_send_author]; 
else 
echo $myrow _ta["tb_send_user"]; 


?> </td> 


<td><?php $query_s=mysql_query("select * from tb_forum_restore where tb_send_id='$myrow_2[tb 


send_id]"); /| 输出 回复 数 以 及 帖子 访问 量 
echo mysql_num_rows($query_s); 


echo "/". 


$query_tn=mysql_query("select * from tb_forum_send where tb_send_id=$myrow_2[tb_send_id]"); 


$myrow_tn=mysql_fetch_array($query_tn); 

echo $myrow_tn["tb_send_count"]; 

?></td> 

</tr> 

<?php ?> 

上 述 讲解 的 是 帖子 主题 浏览 的 实现 方法 ， 下 面 介 绍 帖子 内 容 浏览 功能 的 实现 。 

帖子 内 容 的 输出 是 通过 上 面 提 到 的 send_forum_content.php 文件 来 完成 的 。 在 该 文件 中 输出 帖子 的 
详细 内 容 、 发 帖 人 的 信息 、 回 复 帖 子 的 内 容 和 回复 人 的 信息 ， 并 且 还 对 登录 用 户 进行 了 权限 设置 。 普 
通用 户 只 能 浏览 帖子 的 详细 信息 ， 不 能 进行 其 他 任何 操作 。 

会 员 登 录 后 不 但 可 以 浏览 帖子 的 详细 信息 ， 而 且 可 以 对 帖子 进行 回复 和 引用 ， 收 藏 帖子 、 发 送 短 
信 以 及 加 对 方 为 好 友 ; 如 果 浏 览 的 是 当前 会 员 自 己 发 布 或 者 回复 的 帖子 ， 还 可 以 对 帖子 进行 修改 、 删 
除 和 结 帖 的 操作 。 

管理 员 登 录 后 可 以 执行 上 述 会 员 具 有 的 所 有 操作 ， 并 且 还 可 以 对 帖子 进行 置顶 的 操作 ， 这 是 会 员 
所 不 具备 的 。 

下 面 对 send_forum_content.php 文件 进行 分 步 讲 解 ， 看 看 各 个 部 分 的 功能 都 是 如 何 实现 的 。 


(1) 初始 化 Session 变量 ， 连 接 数 据 库 ， 通 过 include 语句 调用 包含 文件 ， 通 过 $_SERVER 预定 义 


变量 获取 当 


@® 


前 页 面 的 完整 连接 地 址 ， 用 于 实现 帖子 收藏 的 功能 。 有 关 帖 子 收藏 技术 的 讲解 请 参考 7.2.4 
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节 中 的 内 容 。 
(2) 输出 发 帖 人 的 信息 和 发 布 的 帖子 信息 ， 并 且 为 发 送 短信 、 加 为 好 友 、 结 帖 、 置 项 、 修 改 、 删 
除 和 回复 的 操作 设置 超 链 接 。 发 帖 信息 文件 是 send_forum_content.php， 其 关键 代码 如 下 : 


<!-- 从 数据 库 中 读 取出 指定 帖子 的 发 布 人 的 信息 --> 


<?php 

$query_1=mysql_query("select * from tb_forum_send where tb_send id='$_GET[send id]",$conn); 
$myrow_1=mysql_fetch_array($query_1); // 读 取 指 定 帖子 
$query_2=mysql_query("select * from tb_forum_user where tb_forum_user='$myrow_1[tb_send_user]",$conn); 
$myrow_2=mysql_fetch_array($query_2); // 从 会 员 信息 表 中 读 取 指定 的 会 员 信 息 
echo "<img src='$myrow_2[tb_forum_pictureJ'>"."<br>";”// 输 出 图 片 

echo "发 帖 人 :"; 

echo $myrow_2["tb_forum_user"]."<br>"; /| 输出 会 员 名 

echo "注册 时 间 :"."<br>"; 

echo $myrow_2["tb forum_date"]."<br>"; /| 输出 注册 时 间 

echo "积分 :"; 

echo $myrow_2["tb_forum_grade"]."<br>"; /| 输出 积分 


if($_SESSION["tb_forum_user"]==trueX{ 

echo "<a href='send_mail.php?receiving_person=$myrow_2["tb_forum_user"]&&sender=$_SESSION["tb_forum_ 
user] target=' blank><img src='images/index_8.jpg' width='76' height='24' border='0'></a>"."<br>"; 

echo "<a href='my_friend.php?friend=$myrow_2["tb_ forum_user]&&my=$_SESSION["tb_ forum_user"] target= 
”blank><img src='images/index_8 (1).jpg' width='82' height='24' border='0'></a>"; 


人 
<!-- 从 数据 库 中 读 取 出 指定 帖子 的 信息 --> 
<?php if($myrow_1["tb_forum_end"]!=1){ ?> /| 判断 帖子 是 否 已 经 结 帖 


<a href="end_forum.php?send id=<?php echo $_GET[send id];?>&send user=<?php echo S$myrow_3 
["tb_send_ user"];?>"> 结 帖 </a> 
<?php ”J}else{f ”echo" 已 结 帖 ; } ?> 
</td></tr> 
<tr><td height="21"><?php echo $myrow_3["tb_send_content"]; ?></td></tr> 
<!-- 为 不 同 功能 的 实现 设置 超 链接 --> 
<tr><td align="right"> 
<?php if($myrow_3["tb_send_accessories"]==truejf{ /| 判断 是 否 存在 附件 

echo "<a href='download.php?accessories=$myrow_3["tb_send_accessories"]'> 附 件 </a>";}?>&nbsp;&nbsp; 
&nbsp;&nbsp; 

<a href="permute_send.php?permute_id=<?php echo $myrow_3["tb_send_id"];?>"> 和 置顶 </a>、 

<a href="recompose_send.php?recompose_id=<?php echo $myrow_3["tb_send_id"];?>&&recompose_user= 
<?php echo $myrow_3["tb_send_user"];?>"> 修 改 </a>、 

<a href="delete_send.php?delete_id=<?php echo $myrow_3["tb_send _id"];?>&&delete_user=<?php echo 
$myrow_3["tb_send_user"];?>"> 删 除 </a>、 

<a href="send_forum_content.php?send_big_type=<?php echo $_GET ["send_ big _type"];?>&&send_ 
small_type=<?php echo $_GETI["send_small_type"];?>&&send_id=<?php echo $_GET["send_id"];?>#bottom"> 
回复 </a></td> 

</tr> 


(3) 输出 与 该 帖子 相关 的 回复 帖子 的 信息 ， 以 及 回复 人 的 信息 ， 同 样 也 为 发 送 短信 、 加 为 好 友 、 
引用 、 修 改 和 删除 操作 设置 了 超 链接 。 其 中 在 输出 回复 帖子 的 内 容 时 ， 还 对 回帖 内 容 进 行 判断 ， 判 断 
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该 帖子 是 否 被 管理 员 屏蔽 。 

实现 的 方法 与 第 〈2) 步 相 同 ， 这 里 不 再 歼 述 ， 完 整 代码 请 参考 本 书 附 带 光 盘 中 的 内 容 。 

(4) 积分 排行 功能 的 实现 主要 就 是 从 会 员 信息 表 中 读 取 会 员 的 积分 数据 ， 并 且 按 照 降 朝 排列 ， 输 
出 积分 最 高 的 前 10 名 用 户 。 程 序 代码 如 下 : 


<?php 
$sql=mysql_query("select tb_forum_user,tb_forum_grade from tb_forum_user order by tb_forum_grade desc 
limit 10"); 
/从 数据 库 中 读 取出 积分 最 高 的 10 个 用 户 

while($myrow=mysql_fetch_array($sql)){ // 循 环 输出 用 户 姓 名 和 积分 
?> 
<tr><td width="45%" height="19" align= "right"> 

<a _ href="person_data.php?person_id=<?php echo S$myrow[tb_forum_user];?>"><?php echo $myrow 
[tb_forum_user];?></a>&nbsp;</td> 

<td width="55%" align="left"> 一 一 <?php echo $myrow[tb_ forum_grade];?></td> 

</tr> 
<?php }?> 


(5) send_forum_content.php 文件 的 第 5 部 分 是 一 个 form 表单 ， 用 于 提交 回复 帖子 的 信息 。 有 关 
回复 帖子 的 操作 将 在 7.4.4 节 中 做 详细 介绍 。 


7.4.4 帖子 回复 功能 实现 


帖子 回复 中 提交 的 form 表单 存储 在 7.4.3 节 中 介绍 的 send_forum_content.php 文件 中 。 

在 这 个 form 表单 中 ,将 回复 主题 附件、 回复 内 容 、 发 布 帖子 id 和 回复 人 信息 都 提交 到 send_forum_ 
content_ok.php 文件 中 进行 处 理 ， 完 成 帖子 的 回复 。 

在 对 回复 内 容 进行 编辑 时 还 应 用 了 UBB 技术 , 实现 对 回复 的 内 容 进行 编辑 ,并且 还 对 回复 内 容 的 
字 节 数 进行 了 控制 。 

UBB 技术 的 实现 是 通过 UBBCode.js 文件 来 完成 的 ， 该 文件 存储 于 根 目录 下 的 js 文件 夹 中 ， 限 制 
和 统计 回复 内 容 字 节 数 的 方法 是 通过 textjs 文件 来 完成 的 , 该 文件 同样 存储 于 根 目 录 下 的 js 文件 夹 中 。 
form 表单 存在 于 send_forum_content.php 文件 中 ， 其 关键 代码 如 下 : 


<form action="send_forum_content_ok.php" method="post" enctype="multipart/form-data” name="myform"> 
<tr><a name="bottom" id="bottom"></a><! 一 定义 命名 锚 记 一 > 
<!-- 输 出 引用 的 回帖 的 标题 -> 
<td width="641"><input name="restore_subject" type="text" id="restore_subject" value=" 
<?php 
if($_GET[T"cite"]==true){ 
// 从 数据 库 中 获取 回复 帖子 的 标题 
$query=mysql_query("select * from tb_forum_restore where tb_restore_id='$_GETIcite]"); 
$result=mysql_fetch_array($query); 
echo "摘自 (".$result["tb_restore_user"].") :".$resultftb_restore_subject]; 。“// 输 出 引用 标题 
上 
?>" size="80"> 
<input type="hidden" name="tag" value="<?php echo $myrow_3["tb_forum_end"];?>"></td> 
</tr> 


&. 


ok.php 文件 。 


第 7 章 论坛 模块 (PHP+MySQL 实现 ) 


<tr><td height="30" align="right"> 文 字 编 程 区 : </td> 
<td width="641"> 
<img src="images/UBB/B.gif" width="21" height="20" onClick="bold()">&nbsp; 
<img src="images/UBB/I.gif" width="21" height="20" onClick="italicize()"> 字 体 
<select name="font"” class="wenbenkuang” id="font” onChange=" showfont(this.options [this. 
selectedIndex].value)"> 
<option value=" 宋 体 " selected> 宋 体 </option> 
<option value=" 黑 体 "> 黑体 </option> 
</select></td> 
</tr> 
<tr> 
<td align="right"> 文 章 内 容 : </td> 
<td width="641">< 上 -统计 回复 内 容 ， 输 出 引用 的 回复 内 容 --> 
<textarea name="file" cols="80" rows="10" id="file" onKeyDown="countstrbyte(this.form.file,this.form. 
total,this.form.used ,this.form.remain);” onKeyUp="countstrbyte (this.form .file,this.form.total,this.form.used,this. 
form.remain);"> 
<?php 
if($_GETI"cite"]==trueX{ 
// 输 出 回复 帖子 的 内 容 
$query=mysql_query("select * from tb_forum_restore where tb_restore_id='$_GETIcite]"); 
$result=mysql_fetch_array($query); 


echo $result[tb_restore_content]; // 输 出 引用 的 回复 内 容 
} 
?> 
</textarea> 
<input type="hidden" name="tb_send_id" value="<?php echo $_GET["send_id"];?>"> 
<input type="hidden" name="tb_restore_user" value="<?php echo $_SESSION["tb_forum_user];?>"> 
</td> 
</tr> 
<tr> 


<td height="25" colspan="2">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
<input name="submit" type="submit" id="submit" value=" 提 交 " onClick="retum check();"> 

最 大 字 节 数 : <input type="text" name="total" disabled="disabled" class="textbox" id="total” value="500" 
Size="5"> 

输入 : <input type="text" name="used" disabled="disabled" class="textbox” id="used" value="0" size="5"> 

<input name="reset" type="reset" id="reset" value=" 重 写 "></td> 
</tr> 
</form> 


至 此 ， 回 复 帖子 的 提交 文件 的 内 容 讲解 完毕 ， 接 下 来 介绍 回复 帖子 的 处 理 页 send_forum_content_ 


在 该 文件 中 实现 对 回复 帖子 中 提交 的 数据 进行 存储 ， 并 且 更 新 帖子 的 回复 次 数 ， 将 最 后 回复 的 会 


员 名 和 回复 时 间 更 新 到 tb_forum_send 表 中 的 tb_send_author 和 tb_send_ltime 字段 下 ， 实 现在 论坛 浏览 
页 面 显示 最 后 回复 该 帖子 的 会 员 名 ， 并 在 浏览 页 面 将 最 新 回复 的 帖子 移 置 所 有 帖子 最 顶端 ， 同 时 将 发 
布 帖子 数据 表 〈tb_forum_send) 中 的 tb_send_types 字段 更 新 为 1， 表 明 该 帖子 已 经 有 回帖 。 


中 是 否 包含 附件 ， 如 果 不 存在 附件 ， 则 直接 将 获取 的 数据 添加 到 指定 的 数据 表 中 ， 并 且 更 新 帖子 的 回 


在 send_forum_content_ok.php 文件 中 ， 首 先 获取 到 form 表单 中 提交 的 数据 ， 然 后 判断 回复 的 内 容 


复 次 数 和 发 布 帖子 数据 表 中 tb_send_types 字段 的 值 为 1。 


如 果 存 在 附件 ， 而 附件 的 大 小 超过 上 传 文件 大 小 的 限制 ， 则 将 给 出 提示 信息 “上 传 文件 超过 指定 


_ 国 
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赤 可 5 ”5 

如 果 存 在 附件 ， 并 且 在 指定 的 范围 之 内 ， 则 先 将 该 附件 存储 到 服务 器 中 指定 的 文件 夹 下 ， 然 后 再 
将 附件 在 服务 器 中 的 存储 路 径 和 其 他 数据 一 起 存储 到 指定 的 数据 表 中 ， 同 样 也 更 新 帖子 的 回复 次 数 和 
发 布 帖子 数据 表 中 tb_send_types 字段 的 值 为 1。 该 过 程 是 在 send_forum_content_ok.php 文件 中 实现 的 ， 
相关 程序 代码 如 下 : 


<?php SESSION_start(); include_once("conn/conn.php"); 


if($_SESSION["tb_forum_user"]==trueX{ /判断 是 否 是 正确 登录 
$tb_restore_subject=$_POST["restore_subject"]; /获取 回复 帖子 的 主题 
$tb_restore_content=$_POST['file"]; /获取 上 传 的 附件 
$tb_restore_user=$_POST["tb_restore_user"]; /获取 回复 人 
$tb_send_id=$_POST["tb_send id"]; /获取 要 回复 帖子 的 id 
$tb_restore_date=date("Y-m-d 3 // 定 义 回复 时 间 
$tb_send_date=date("Y-m-d H:i:s"); // 定 义 最 后 回复 时 间 
$cite=$_POST["cite"]; /定义 引用 值 
if($_FILES["restore_accessories"]["size"]==0X{ /| 判断 是 否 有 附件 上 传 


if(mysql_query("insert into tb forum_restore (tb_restore subject,tb_ restore_content,tb_restore user,tb_ 
send id,tb_restore_date) values (".$tb_restore_ subject.",".$tb_ restore_content.”,". $tb_restore_user.",". 
S$tb_send_id.",".$tb_restore_date.”)",$conn)X{ 
// 更 新 最 后 发 布 者 和 最 后 回复 时 间 的 操作 
mysql_query("update tb_forum_send set tb_forum_send.tb_send_author='$tb_restore_user where tb_send_id 
= $tb_send_id"); 
mysql_query("update tb_forum_send set tb_forum_send.tb_send_ltime='$tb_send_date' where tb_send id = 
S$tb_send_id"); 
// 执 行 更 新 回复 次 数 的 操作 
mysql_query("update tb_forum_restore set tb_forum_counts=tb_forum_counts+1",$conn); 
mysql_query("update tb_forum_send settb_send_types=1 where tb_send_id='$tb_send_id",$conn); 
echo "<script>alert(' 回 复 成 功 !');history.back();</script>"; 
mysql_close($conn); 
Jelse{ 
echo "<script>alert( 回 复 失败 !");history.back();</script>"; 
mysql_close($conn); 


} 


} 
if($_FILES["restore_accessories"]["size"] > 20000000X // 判 断 上 传 附件 是 否 超过 规定 文件 的 大 小 
echo "<script>alert(' 上 传 文件 超过 指定 大 小 ! ');history.go(-1);</script>"; 
exit(); 
}else{ 
$path = '.file/.time().$_FILES[restore_accessories][name'];， /定义 上 传 文件 名 称 和 存储 的 路 径 
// 将 附件 存储 到 服务 器 指定 的 文件 夹 下 
if (move_uploaded _file($_FILES[restore_accessories][tmp_name'"],$path)) {// 执 行 存储 操作 
if(mysql_query("insert into tb_forum_restore (tb_restore_subject, tb_restore_content,tb_restore_ user,tb_ 
send id,tb_restore_date,tb_restore_accessories) values (".$tb_restore_ subject.",".$tb_restore_ content.",".$tb_ 
restore_ user.",".$tb_send_id.",".$tb_restore_date.",".$path.™")", 
$conn)){ /执行 添加 操作 
mysql_query("update tb_forum_send set tb_ forum_send.tb_send_author='$tb_restore_user where 
tb_send_ id = $tb_send_ id"); 
mysql_query("update tb _forum_send set tb forum_send.tb_send_ltime='$tb_send_date' where tb_ 
@ 
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send id = $tb_send _id"); 
mysql_query("update tb_forum_restore settb forum_counts=tb_forum_counts+1",$conn); 
mysql_query("update tb_forum_send set tb_send_types=1 where tb_send_ id='$tb_send_ id",$conn); 
echo "<script>alert(' 回 复 成 功 !");history.back();</script>"; 
mysql_close($conn); 
Jelse{ 
echo "<script>alert(' 回 复 失败 !");history.back();</script>"; 
mysql_close($conn); 


» 
}else{ 
echo "<script>alert( 对 不 起 ， 您 不 可 以 回复 帖子 ， 请 先 登 录 到 本 站 ');history.back();</script>"; 


} 
?> 


在 论坛 显示 页 面 中 ， 显 示 的 各 个 主题 内 容 是 按照 最 后 回复 的 次 序 倒序 排序 的 。 这 样 可 以 使 用 户 最 
后 回复 的 帖子 始终 处 于 最 上 方 ， 使 其 他 用 户 能 够 第 一 时 间 浏 览 到 这 个 帖子 以 及 回复 内 容 。 这 一 设 定 本 
来 是 为 了 方便 用 户 能 够 及 时 发 现 最 新 的 帖子 ， 但 是 同时 也 会 导致 一 些 问 题 ， 会 有 一 些 会 员 回复 很 久 以 
前 的 帖子 ， 这 样 老 主题 就 会 显示 在 页 面 最 上 方 ， 这 种 恶意 回复 帖子 的 行为 被 成 为 “ 据 莫 ”。 为 了 防止 
这 种 用 户 恶意 行为 ， 解 决 该 问题 的 办 法 是 在 该 提交 处 理 页 也 添加 一 段 代 码 来 防止 这 种 “ 据 墓 ”行为 ， 
相关 代码 如 下 : 

$query_k=mysql_query("select * fom tb_forum_send where tb_send_id='$tb_send_id"); /从 数据 库 中 提取 发 帖 时 间 

$myrow_k=mysql_fetch_array($query_k); 


$newtime=substr($myrow_k["tb_send_date"],0,10); // 把 发 帖 的 时 间 截 取 为 “年 -月 -日 ”的 形式 
$array=explode("-",$newtime); /用 “-” 做 分 隔 符 截取 时 间 
$year=date("Y"); // 获 取 当 前 时 间 


$month=date("m"); 
$day=date("d"); 
$sendtime=($year-$array[0]-1)*365+abs(($day-$array[1]))*30+abs($day-$array[2]);”// 计 算 已 发 帖 时 间 


if($sendtime>180) // 如 果 发 帖 时 间 已 经 超过 180 天 
echo "<script>alert(' 该 帖 发 布 时 间距 今 已 超过 半年 ， 不 能 回复 !");history.back();</script>"; // 则 不 能 进行 回复 
else{ 

// 执 行 回复 代码 

} 


至 此 ， 帖 子 回复 功能 讲解 完毕 ， 详 细 的 程序 代码 可 以 参看 本 书 附带 光盘 中 的 内 容 。 
7.5 帖子 搜索 


7.5.1 帖子 搜索 概述 
帖子 搜索 是 指 在 论坛 中 按照 指定 的 关键 字 ， 从 论坛 发 布 的 帖子 和 回复 的 帖子 中 查询 出 符合 条 件 的 


@ 
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基于 论坛 较 高 的 时 效 性 ， 一 些 大 中 型 热门 论坛 每 天 会 被 大 量 的 用 户 访问 、 发 帖 、 回 复 ， 这 同样 也 
给 浏览 者 在 帖子 查找 上 带 来 诸多 不 便 。 为 了 查找 相关 内 容 而 浏览 每 一 页 上 的 所 有 帖子 是 不 现实 的 ， 而 


帖子 搜索 功能 即 可 满足 浏览 者 的 要 求 ， 迅 速 地 查找 到 与 浏览 者 提交 的 关键 字 有 关 的 内 容 。 


在 论坛 模块 中 ,帖子 搜索 从 content.php 文件 中 设置 的 帖子 搜索 文本 框 开始 (如 图 7.28 所 示 ) ， 将 要 搜 
索 的 关键 字 提 交 到 search.php 文件 中 ， 在 该 文件 中 执行 模糊 查询 ， 并 将 查询 的 结果 输出 ， 如 图 7.29 所 示 。 


最 新 子 | |[ 精 中 区 | [ 热 # 区 | [站 名] hp 。 视 案 5I 沁 


公 告 】 程 序 测试 
置 项 】 在 Pip 中 ,如何 控制 上 传 文件 的 大 小 ?了 
置 项 】 Snarty 是 什么 ? 


图 7.28 提交 搜索 关键 字 
7.5.2 ”帖子 搜索 功能 实现 


帖子 搜索 通过 论坛 首页 的 搜索 文本 框 提交 关键 字 ， 对 站 内 的 所 有 帖子 执行 搜索 操作 ， 将 搜索 得 到 
的 帖子 标题 名 称 、 发 帖 人 、 回 复 次 数 等 信息 以 超 链 接 的 形式 在 搜索 显示 页 中 输出 。 另 外 ， 一 些 与 关键 


局 
2008-08-0! ls:5136 11 3 
2008-08-0! 18;52:24 。 1 
2008-08-0! 19:50.31 11 o 
2008-08-0! 19:51:14 11 
2008-0T-21 17:0428 11 1 
2008-08-01 19:49:28 11 1 
2008-08-0! 19:48:49 。 11 1 
页: ML w 
同时 间 Mg 人 同 复 次 数 
2005-08-01 11:13:45 11 2 
Smarty 是 一 php 民权 3 学 2008-08-02 08:36:36 。 请 训 毕 1 
PB 中 控 机 上 传 文件 的 大 ; 2008-08-01 11:13:45 11 2 
Saazty 是 一 个 php 异 上 3 降 2008-03-02 08:35:36 。 濡 让 侣 1 
帖 了 统计 :2 条 :WL x 


图 7.29 ”帖子 搜索 的 结果 


字 有 关 的 帖子 ， 也 会 以 超 链接 形式 在 发 布 的 帖子 下 方 输出 。 帖 子 搜索 的 实现 过 程 如 下 : 


(1) 在 content.php 文件 中 ， 首 先 创建 一 个 form 表单 ， 提 交 帖 子 搜索 的 关键 字 ， 将 关键 字 提 交 到 


search.php 文件 中 。content.php 文件 中 的 关键 代码 如 下 : 


<form name="form1” method="post” action="search.php?class= 搜索 引擎 &&content=<?php echo $_GET 


["content"];?> &&content_1=<?php echo $_GET["content_1"];?>" onSubmit="return check_submit();"> 


<tr> 


<td><input name="tb_send_subject_content" type="text" size="20" />&nbsp;&nbsp;</td> 


<td width="25%" rowspan="2"><input type="image" name="imageField" src="images/index_71.jpg" /></td> 


</tr> 
</form> 


(2) 创建 search.php 文件 ， 根 据 表单 中 提交 


询 ， 将 查询 的 结果 以 分 页 的 形式 输出 到 页 面 中 。 


<?php session_start(); include("conn/conn.php"); 
if($page==""X{ $page=1;} 

if($pages=="){ $pages=1; } 

if($link_type==""X S$link_type=0; } 
if($link_types==""X $link_types=0; } 
$content=$_GET["content"]; 


@ 


其 中 模糊 查询 的 关键 代码 如 下 : 
/初始 化 Session 变量 ， 连 接 数 据 库 


/判断 变量 的 值 是 否 为 空 ， 用 于 分 页 显示 
// 判 断 变量 的 值 是 否 为 空 ， 用 于 分 页 显示 


// 获 取 帖 子 的 类 型 


的 关键 字 ， 分 别 在 发 布 帖 子 和 回复 帖子 中 执行 模糊 查 
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$content_1=$_GET["content_1"]; 1/ 获取 帖 子 的 类 别 

// 从 发 布 的 帖子 中 查询 

$query_6=mysql_query("select * from tb_forum_send where tb_send_subject like '%$tb_send_subject_content%' 
ortb_send_content like '%$tb_send_subject_content%"); 

// 从 回复 的 帖子 中 搜索 

$query_7=mysql_query("select * from tb_forum_restore where tb_restore_subject like '%$tb_send_ subject_ 
content%' or tb_restore_content like '%$tb_send_subject_content%"); 


// 统 计 查 询 的 结果 
if(mysql_num_rows($query_6)>0 or mysql_num_rows($query_7)>0 上 
if($pageX{ // 定 义 分 页 的 变量 
$page_size=10; /定义 每 页 显示 的 数量 


$query="select count(*) as total from tb forum_send where tb_send_subject like '%$tb_send_subject_ 
content%' or tb_send_content like '%$tb_send_subject_content%"; 
$result=mysql_query($query); /执行 查询 语句 
$message_count=mysql_result($result,0,"total"); // 获 取 查 询 结 果 
$page_count=ceil($message_count/$page_size); 。“”// 计 算 共 有 几 页 
S$offset=($page-1)*$page_size; // 获 取 上 一 页 的 最 后 一 条 记录 
$query_2=mysql_query("select * from tb _ forum_send where tb_send subject like '%$tb_send_ 
Subject_content%' or tb_send_content like '"%$tb_send_subject_content%' limit $offset, $page_size"); 
?> 


至 此 ， 帖 子 搜索 功能 讲解 完毕 ， 有 关 查 询 结果 的 分 页 输出 ， 这 里 不 做 介绍 ， 完 整 代码 可 以 参考 本 
附带 光盘 中 的 内 容 。 


7.6 帖子 管理 


7.6.1 帖子 管理 概述 


对 论坛 帖子 的 管理 ， 已 成 为 论坛 维护 必 不 可 少 的 一 部 分 ， 帖 子 管理 甚至 可 以 直接 影响 到 用 户 的 访 
问 和 回帖 数量 ， 其 中 帖子 管理 包括 帖子 的 结 帖 功能 、 优 秀 帖子 的 置顶 操作 、 设 置 帖子 的 类 别 以 及 项 帖 
管理 的 实现 等 。 
结 帖 功 能 是 对 会 员 自己 发 布 的 帖子 进行 操作 ， 当 获取 到 满意 的 答案 之 后 ， 就 可 以 对 帖子 进行 结 帖 
操作 ， 一 旦 结 帖 之 后 就 不 可 以 再 对 该 帖 进行 回复 。 其 运行 结果 如 图 7.30 所 示 。 
什么 是 smarty? 添加 到 我 的 收藏 夫 


居间 :2011-01-14 08 .34 人 9 各 主 ( 匡 扣 ) 
人 ee 


单 击 结 帖 


置顶 修改 、 册 除 、 回 各 


图 7.30 结 帖 功能 的 运行 结果 
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结 帖 功能 避免 了 在 论坛 中 对 一 个 帖子 无 休止 地 回复 ， 浪 费 系统 资源 ， 同 时 也 确保 了 论坛 中 帖子 的 
规范 性 。 

帖子 管理 的 另 一 功能 是 帖子 的 分 类 输出 。 在 论坛 中 ， 根 据 帖 子 的 发 布 时 间 、 帖 子 内 容 的 特殊 性 以 
及 受 关注 的 程度 ， 还 有 帖子 是 否 有 人 回复 等 ， 对 帖子 进行 了 分 类 处 理 ， 分 为 最 新 帖子 、 精 华 区 、 热 点 
区 和 待 回复 等 儿 个 类 别 。 其 运行 结果 如 图 7.31 所 示 。 


MINGRISOFT 实时 升级 服务 


确 程 著 之 亲 
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图 7.31 帖子 的 分 类 输出 

顶 帖 管理 是 针对 管理 员 的 帖子 置顶 权限 而 设置 的 ， 因 为 不 可 能 将 某 个 帖子 永远 置顶 ， 所 以 创建 了 

顶 帖 管理 这 个 功能 。 该 功能 存在 于 论坛 的 后 台 管 理 中 ， 实 现 帖子 置顶 和 取消 置顶 的 操作 。 其 运行 结果 
如 图 7.32 所 示 。 


实时 升级 服务 /ON 


eu 再 万 上 开发 要 ， 必 英和 天 新 沁 认 


所 作 
2000-0001 9 1 | 
oom-00-01 1928 1 可 区 可 


ET [| 
oo 0 | 


并: I 页 慎 [ 


图 7.32 顶 帖 管理 功能 的 运行 结果 


7.6.2 ” 结 帖 功能 实现 


论坛 的 管理 员 也 具备 这 个 权限 ， 可 以 根据 帖子 的 回复 情况 ， 在 确定 已 经 有 满意 答案 的 情况 下 ， 而 
帖子 发 布 人 又 没有 进行 结 帖 操作 的 ， 可 以 由 管理 员 来 执行 这 项 操作 。 管 理 员 的 结 帖 操作 是 在 论坛 后 台 
管理 的 帖子 管理 中 完成 的 。 

帖子 是 否 已 经 结 帖 是 根据 帖子 在 数据 表 中 也 _forum_end 字段 的 值 来 判断 的 ， 如 果 字段 的 值 为 1， 
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则 说 明 帖 子 已 经 结 帖 ， 否 则 没有 结 帖 。 所 以 结 帖 操 作 就 是 将 指定 帖子 在 数据 表 中 的 tb_forum_end 字段 
的 值 更 新 为 1。 

在 论坛 模块 中 ， 结 帖 操作 是 通过 在 send_forum_content.php 文件 中 设置 的 一 个 “ 结 帖 ” 的 超 链接 来 
执行 的 。 通 过 “ 结 帖 ” 这 个 超 链接 ， 链 接 到 end_forum.php 文件 ， 在 这 个 文件 中 根据 传递 的 id 值 ， 执 
行 更 新 指定 帖子 tb_forum_end 字段 值 的 操作 。 

在 send_forum_content.php 文件 中 设置 “ 结 帖 ” 超 链接 ， 并 且 根 据 tb_forum_end 字段 的 值 来 判断 
输出 的 内 容 。 关 键 代 码 如 下 : 

<?php 

if($myrow_1["tb_forum_end"]!=1){ /判断 如 果 字 段 的 值 不 为 1， 则 输出 下 面 的 链接 

?> 

<a href="end_forum.php?send id=<?php echo $_GET[send id];?>&send_user=<?php echo $myrow_3[tb_ 

send_user];?>"> 结 帖 </a> <!-- 设 置 结 帖 操 作 的 超 链接 --> 

<?php 

}else{ 

echo "已 结 帖 "; 

} 


?> 


在 end_forum.php 文件 中 ， 执 行 结 帖 的 操作 ， 以 $send_id 变量 传递 的 帖子 id 值 为 依据 。 程 序 代码 
如 下 : 


<?php session_start(); include("conn/conn.php"); 
if($_GET["send_id"]==true and $_GET["send_user"]==$_SESSION["tb_forum_user"]X{ 
$result=mysql_query("update tb_forum_send settb forum_end='1 where tb_send id=".$_GET[send_id].""); 
if($result==trueX{ 
echo "<script>alert(' 结 帖 激 活 !"); history.back();</script>"; 


} 
}else{ 


} 


?> 


7.6.3 ”设置 帖子 类 别 


精华 区 、 热 点 区 和 待 回复 这 3 个 类 别 的 实现 方法 相同 ， 都 是 根据 数据 库 中 帖子 指定 的 字段 值 进 行 
判断 ， 精 华 区 根据 字段 tb_send_type_distillate 的 值 判断 ;热点 区 根据 字段 tb_send_ type_hotspot 的 值 
判断 ; 而 待 回复 则 根据 字段 tb_send_ types 的 值 判断 。 这 里 以 精华 区 帖子 的 输出 为 例 ， 该 输出 文件 
distillate.php 的 关键 代码 如 下 : 


echo "<script>alert(' 您 不 具备 该 权限 1"); history.back();</script>"; 


<?php 
if($pageX{ // 实 现 精华 区 帖子 的 分 页 输出 
$page_size=10; /| 每 页 显示 10 条 记录 


// 以 tb_send_type_distillate 字段 的 值 是 否 为 1 为 条 件 ， 如 果 为 1 则 是 精华 帖子 ， 否 则 不 是 
$query="select count(*) as total from tb_forum_send where tb_send_small_ type='$_GET[content_1]' and 


tb_send_type_distillate=1"; 
® 
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$result=mysql_query($query); 
$message_count=mysql_result($result,0,"total"); 
S$page_count=ceil($message_count/$page_size); 
S$offset=($page-1)*$page_size; 
$query_2=mysql_query("select * from tb forum send where tb_send_ small type='$content_1' and 
tb_send type_distillate='1' limit $offset, $page_size"); 
while($myrow_2=mysql_fetch_array($query_2)X{ 
?> 


上 述 介 绍 的 是 如 何 从 数据 库 中 获取 到 指定 类 别 的 帖子 ， 下 面 讲解 这 些 帖 子 的 类 别 是 如 何 设 置 的 。 
最 新 帖子 不 需要 任何 设置 ， 只 要 是 帖子 发 布 之 后 ， 自 动 会 生成 一 个 id 值 ， 根 据 id 值 自动 可 以 读 取 
到 最 新 的 帖子 。 
而 精华 区 、 热 点 区 和 待 回复 则 都 需要 设置 。 其 中 ， 设 置 精华 区 和 热点 区 帖子 的 方法 相同 ， 都 是 在 
论坛 的 后 台 管 理 中 进行 操作 ， 通 过 form 表单 创建 复 选 框 ， 将 指定 帖子 的 tb_send_type_distillate 或 者 
tb_send_type_hotspot 字段 的 值 设置 为 1， 运 行 结果 如 图 7.33 所 示 。 
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图 7.33 设置 帖子 的 类 别 


帖子 类 别 的 设置 操作 通过 两 个 文件 完成 : 一 个 是 update_forum.php, 用 于 提交 要 设置 类 别 帖 子 的 id; 
另 一 个 是 update_forum_ok.php， 根 据 提交 的 id 值 执行 设置 帖子 类 别 的 操作 。 

在 update_forum.php 文件 中 ， 首 先 创建 一 个 form 表单 ， 从 数据 库 中 读 取 帖子 的 数据 ， 并 且 为 每 个 
帖子 设置 一 个 复 选 框 ， 复 选 框 的 值 是 帖子 的 id， 再 分 别 创建 精华 帖 和 热点 帖子 的 “提交 ”按钮 ， 同 时 
也 创建 取消 帖子 类 别 的 按钮 ; 最 后 将 数据 提交 到 update_forum_ok.php 文件 中 。 其 中 ， update_forum.php 
文件 的 关键 代码 如 下 : 


<form name="form1" method="post" action="update_forum_ok.php"> 
<tr> 
<td height="25" align="center” class="STYLE1"><input name="<?php echo $myrow["tb_send id"];?>" 
type=" checkbox" value="<?php echo $myrow["tb_send _id"];?>"></td> 
<td align="center" class="STYLE1"><?php echo $myrow["tb_send_type_distillate"];?></td> 
<td align="center" class="STYLE1"><?php echo $myrow["tb_send_type_hotspot"];?></td> 
</tr> 


<tr> 


@ 
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<td align="center"><input type="submit" name="Submit" value=" 精 华 帖 "> 
<input type="submit" name="Submit3" value=" 取 消 "></td> 

<td align="center"><input type="submit" name="Submit2" value=" 热 门 帖 "> 
<input type="submit" name="Submit4" value=" 取 消 "></td> 

<td colspan="3" align="center"><input type="submit" name="Submit5" value=" 结 帖 "> 
<input type="submit" name="Submit6" value=" 取 消 "></td> 

</tr> 

</form> 


在 update_forum_ok.php 文件 中 ,根据 表单 中 提交 的 帖子 的 id 值 ， 通 过 while 语句 和 list0) 函 数 ， 循 
环 读 取 表 单 中 提交 的 帖子 的 id 值 ， 执 行 设置 帖子 类 别 和 取消 帖子 类 别 的 操作 。 关 键 代码 如 下 : 


<?php session_start(); include("conn/conn.php"); 

if(lempty($_POSTI"Submit"])) /| 判断 从 表单 获取 的 Submit 是 否 为 空 
$Submit=$_POST["Submit"]; // 如 果 不 为 空 ， 则 将 值 赋 给 变量 $Sumbit 
if(lempty($_POST["Submit2"])) 

$Submit2=$_POST["Submit2"]; 

if(lempty($_POST["Submit3"])) 

$Submit3=$_POST["Submit3"] 

if(lempty($_POST["Submit4"])) 

$Submit4=$_POST["Submit4"] 

if(lempty($_POST["Submit5"])) 

$Submit5=$_POST["Submit5"]; 

if(lempty($_POST["Submit6"])) 

$Submit6=$_POST["Submit6"]; 


if($Submit==" 精 华 帖 "){ // 刹 断 当 表单 提交 的 值 等 于 精华 帖 时 执行 下 面 的 操作 
while(list($name,$value)=each($_POST)){ /通过 list() 函 数 获取 表单 提交 的 值 ， 并 循环 输出 内 容 
$result=mysql_query("update tb _ forum_send set tb_send type_distillate='1' where tb_send id= 
".$name."); /执行 设置 精华 帖 的 操作 


if($result==trueX{ 
echo "<script>alert(' 精 华 帖 激活 成 功 !"); window.location.href='index.php?title= 帖 子 管理 ';</script>";}} 


3 
if($Submit3==" 取 消 "){ /执行 取消 精华 帖 的 设置 
whilellist($name,$value)=each($_POST)) 
$result=mysql_query("update tb_forum_send set tb_send type_distillate='0' where tb_send id= 
".$name."™"); 
if($result==trueX{ 
echo "<script>alert(' 精 华 帖 取消 !); window.location.href='index.php?title= 帖 子 管理 ;</script>";}} 
)?> 


人 其 主要 通过 while 循环 语句 和 list()、 
each() 函 数 来 完成 。 


(1) each0) 函 数 返 回 数组 中 当前 指针 位 置 的 键 名 和 对 应 的 值 ， 并 向 前 移动 数组 指针 。 键 值 对 被 返 
回 为 4 个 单元 的 数组 ， 键 名 为 0、1、key 和 value。 单 元 0 和 key 包含 数组 单元 的 键 名 ，1 和 value 包含 
数据 ， 如 果 内 部 指针 越过 了 数组 的 末端 ， 则 函数 返回 false。 该 函数 的 语法 如 下 : 


array each ( array array) 


_ 国 ) 
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参数 array 为 输入 的 数组 。 
(2) listO 函 数 把 数组 中 的 值 赋 给 一 些 变量 。 与 array0 函 数 类 似 , 不 是 真正 的 函数 ,而 是 语言 结构 。 
listO) 函 数 仅 能 用 于 数字 索引 的 数组 ， 并 且 数 字 索 引 从 0 开始 。 该 函数 的 语法 如 下 : 


void list (mixed…) 


参数 mixed 为 被 赋值 的 变量 名 称 。 

(3) 值得 一 提 的 是 ，while(list ($name,$value)=each($_POST)) 这 条 语句 不 能 在 同一 文件 中 被 重复 利 
用 ， 即 在 执行 完 该 操作 循环 体内 容 后 ， 不 能 通过 另 一 条 与 该 语句 并 行 且 相同 的 循环 语句 再 次 执行 ， 这 
样 可 能 会 导致 一 个 非 预期 的 效果 ， 代 码 如 下 : 


<?php 

whilellist($name,$value)=each($ POST) 

echo "这 条 语句 会 被 输出 "; /该 语句 会 被 执行 

下 

while(list($name,$value)=each($_POST)X 

echo "这 条 语句 不 会 被 输出 "; // 该 语句 不 会 被 执行 

人 

而 待 回复 则 是 在 回复 帖子 的 操作 中 完成 的 , 当 回复 帖子 提交 成 功 后 , 执行 更 新 回复 帖子 中 字段 tb_send_ 
types 的 值 为 1， 表 明 该 帖子 已 经 有 回复 ， 有 关 程 序 代 码 可 以 参考 第 7.4.4 节 中 的 内 容 ， 这 里 不 再 袭 述 。 


7.6.4 项 帖 管理 功能 的 实现 
顶 帖 管理 是 针对 管理 员 的 帖子 置顶 权限 而 设置 的 ， 因 为 不 可 能 将 某 个 帖子 永远 置顶 ， 所 以 创建 了 


顶 帖 管理 这 个 功能 。 该 功能 存在 于 论坛 的 后 台 管 理 中 ， 实 现 帖子 置 项 和 取消 置顶 的 操作 。 其 运行 结果 
如 图 7.34 所 示 。 
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7.34 ”项 帖 管理 功能 的 运行 结果 


顶 帖 管理 功能 的 实现 应 用 了 两 个 文件 : 一 个 是 permute_admin.php， 用 于 从 数据 库 中 读 取出 置顶 帖 
子 的 数据 ， 进 行 分 页 输出 ， 并 且 创 建 foum 表单 ， 为 每 个 帖子 设置 一 个 下 拉 列 表 框 ， 实 现 帖子 置顶 和 取 
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消 置 项 的 操作 。 
其 中 ，permute_admin.php 文件 的 关键 代码 如 下 : 
<?php 
if($page){ 

$page_size=5; /| 每 页 显示 5 条 记录 
$query="select count(*) as total from tb_forum_send where tb_send_type=1"; 
$result=mysql_query($query); /执行 查询 操作 
$message_count=mysql_result($result,0,"total"); /获取 总 的 记录 数 
$page_count=ceil($message_count/$page_size); /| 获取 总 的 页 数 


S$offset=($page-1)*$page_size; 
$query=mysql_query("select * from tb_forum_send where tb_send_type=1 order by tb_send_id desc limit 
S$offset, $page_size"); 
while($myrow=mysql_fetch_array($query)}{ /| 循环 输出 查询 结果 
?> 
<tr> 
<td align="center"> 
<form action="update_permute.php?update_id=<?php echo $myrow["tb_send_id"];?>" method="post" name=" 
form1" class="STYLE1"> 
<select name="tb_send_type" id="tb_send_type"> 
<option value="1"> 置 项 </option> 
<option value="0"> 取 消 </option> 
</select> 
<input type="submit" name="Submit" value=" 执 行 "> 
</form></td> 
</tr> 
<?php }}?> 


另 一 个 是 update_permute.php， 在 该 文件 中 ,根据 表单 中 提交 的 值 和 帖子 id 的 值 ， 将 数据 库 中 tb_ 
send_type 字段 的 值 改 为 提交 的 相关 数值 ， 来 实现 帖子 置顶 和 取消 置顶 的 操作 。 关 键 代 码 如 下 : 


<?php include("../conn/conn.php"); 
S$update_id=$_GET["update_id"]; // 获 取 帖 子 的 ID 值 
// 执 行 帖子 置顶 或 者 取消 置顶 的 操作 
$query=mysql_query("update tb_ forum_send set tb_send_ type='$_POST[tb_send_type] where tb_send_id=' 
S$update_id"); 
if($query==trueX{ 
echo "<script>alert(' 更 新 成 功 !");history.back();</script>"; 
}else{ 
echo "<script>alert(' 更 新 失败 !");history.back();</script>";} 
?> 


7.7 个 人 信息 管理 


7.7.1 个 人 信息 管理 概述 


作为 论坛 模块 ， 登 录用 户 个 人 信息 的 显示 也 是 必 不 可 少 的 元 素 之 一 。 在 整个 系统 中 ， 用 户 不 但 可 


_ 国 ) 
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以 通过 个 人 邮箱 收发 站 内 邮件 ， 而 且 可 以 对 个 人 的 站 内 邮件 进行 管理 。 此 外 ， 登 录用 户 还 可 以 通过 帖 
子 上 所 显示 的 个 人 信息 添加 站 内 其 他 会 员 为 好 友 ， 并 通过 个 人 信息 管理 模块 管理 个 人 好 友 名 单 。 
用 户 的 个 人 信息 管理 主要 通过 用 户 个 人 显示 信息 模块 实现 ， 如 图 7.35 所 示 。 


用 户 昵称 : 多 蒙 个 大 信息 今天 是 : 2011-02-26 C1y | 


7.35 用 户 个 人 信息 显示 


用 户 登 录 成 功 后 ， 可 以 看 到 登录 用 户 的 个 人 信息 ， 单 击 “ 我 的 好 友 ” 或 “我 的 信箱 ”， 即 可 跳 转 
至 相应 的 管理 页 面 。 另 外 ， 该 模块 还 包含 了 个 人 信息 更 改 、 我 参与 的 帖子 、 我 的 收藏 等 管理 超 链接 。 
由 于 篇 幅 限制 ， 这 里 只 对 我 的 信箱 和 我 的 好 友 做 详细 讲解 ， 其 他 相关 程序 请 参见 光盘 中 的 内 容 。 


7.7.2 我 的 信箱 管理 


创建 我 的 信箱 从 第 7.4.3 节 send_forum_content.php 文件 中 设置 的 超 链 接 开 始 。 我 的 信箱 模块 链接 
的 是 send_mail.php 文件 ， 在 该 文件 中 完成 我 的 信箱 模块 的 操作 。 其 运行 结果 如 图 7.36 所 示 。 
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图 7.36 ”我 的 信箱 模块 的 运行 结果 
我 的 信箱 模块 中 主要 包括 写 信 、 收 件 箱 和 发 件 箱 3 个 功能 。 这 3 个 功能 都 在 send_mail.php 文件 中 


在 send_mail.php 文件 中 ， 应 用 switch 语句 ， 根 据 栏目 标识 的 变量 值 ， 实 现 不 同 功能 之 间 的 切换 输 
出 。 该 实现 过 程 的 关键 代码 如 下 : 


<table width="812" height="65" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td><?php echo $_GET[sender];?> 您 好 : 您 现在 有 
<?php 
$query=mysql_query("select * from tb_mail_box where tb_receiving_person='$_GET[sender] and tb_mail_ 
type="); 
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$myrow=mysqlL_num_rows($query); 
echo $myrow; 
?> 条 未 读 信息 ! </td> 
</tr> 
<tr> 
<td><a href="send_mail.php?sender=<?php echo $_GET[sender];?>&&mails= 收 件 箱 "> 收 件 箱 </a></td> 
<td><a href="send_mail.php?sender=<?php echo $_GETIsender];?>&&mails= 发 件 箱 "> 发 件 箱 </a></td> 
<td><a href="send_mail.php?sender=<?php echo $_GET[sender];?>&&mails= 写 信 "> 写 信 </a></td> 


</tr> 
</table> 
<?php 
Switch($mails){ 
Case ™: 
include("write_mail.php"); 
break; 
case " 写 信 ": 
include("write_mail.php"); 
break; 
case " 收 件 箱 ": 
include("browse_mail.php"); 
break; 
case "发 件 箱 ": 


include("browse_send_mail.php"); 


break; 


} 


?> 


(1) 写 信 通 过 write_mail.php 和 write_mail_ok.php 文件 完成 。 通 过 write_mail.php 文件 来 创建 form 
表单 ， 提 交 发 送信 息 的 内 容 。 通 过 write_mail_ok.php 文件 来 对 表单 中 提交 的 内 容 进行 处 理 ， 并 且 将 发 


送信 息 存 储 到 指定 的 数据 表 中 作为 发 送 记 录 。 


(2) 收 信 通 过 browse_mail.php、browse_mail_content.php 和 delete_mail.php3 个 文件 来 完成 。 通过 
browse_mail.php 文件 从 数据 库 中 读 取 出 收 到 信息 的 内 容 , 将 信息 内 容 进 行 分 页 输出 , 并 且 设 置 超 链 接 ， 
链接 到 browse_mail_content.php 文件 。 在 browse_mail_content.php 文件 中 查看 信息 的 详细 内 容 。 通 过 
delete_mail.php 文件 ， 实 现 对 收 信箱 中 的 信息 进行 管理 ， 按 照 不 同类 别 的 信件 可 以 执行 不 同 的 操作 ， 如 
系统 邮件 只 能 进行 删除 和 查看 操作 ， 普 通 邮件 则 可 以 进行 查看 、 回 复 、 删 除 操作 ， 好 友 验 证 邮件 可 以 
执行 将 发 送 验证 的 会 员 加 为 好 友 操作 。 其 运行 结果 如 图 7.37 一 图 7.39 所 示 。 


系统 邮件 :对方 已 经 通过 好 友 申 请 验证 》 


adnin 已 经 通过 了 悠 的 好 友 申 请 


来 自 :adnin 一 一 2011-01-13 


验证 信息 :< 你 好 ， 潘 训 华 > 


你 好 ， 潘 新 华 


来 自 :leonsk 一 -一 2011-01-15 


加 为 好 友 


7.37 ”系统 邮件 查看 


7.38 ”验证 邮件 查看 
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邮件 标题 .< 回复 主题 : [I just want te test this system]> 


ok! copy that! 


来 自 :leonsk 一 -一 2011-01-15 
添加 回复 


图 739 普通 邮件 查看 


判别 邮件 类 型 是 通过 tb_mail_box 表 中 的 yanzheng 字段 来 决定 的 ， 通 过 相关 参数 更 新 数据 库 的 相 
关 类 别 ， 然 后 通过 返回 的 相关 字段 名 称 来 判断 站 内 邮件 的 性 质 ， 最 后 通过 判断 的 结果 输出 对 应 的 邮件 
类 别 ， 该 过 程 是 通过 browse_mail.php 文件 实现 的 ， 其 相关 程序 代码 如 下 : 


<?php session_start();include("conn/conn.php"); 
if(lempty($_GET[I"mail_ id"])X 

$query=mysql_query("update tb_mail_box set tb_mail_type=1 where tb_mail_id='$_GETImail_id]"); 
网 
// 省 略 部 分 代码 
<table width="950" height="185" border="0" align="center" cellpadding="0" cellspacing="0" bgcolor="#F7F7FF"> 
<?php $query=mysql_query("select * from tb_mail_box where tb_mail id='$_GET[mail id]"); 

$myrow=mysql_fetch_array($query); 
ea 
<tr> 

<td width="129">&nbsp;</td> 

<td height="35" colspan="2" bgcolor="#FFFFFF"><div class="ttstyle"> 

<div class="title"> 

<span class="fontcolor"> 

<?php 

if($myrow["yanzheng"]==1) // 当 该 字段 值 为 1 时 为 验证 信息 

echo "验证 信息 :"; 

if($myrow["yanzheng"]==2) // 当 该 字段 值 为 2 时 为 普通 邮件 

echo "邮件 标题 :"; 

if($myrow["yanzheng"]==3) // 当 该 字段 值 为 3 时 为 已 验证 邮件 

echo "已 验证 邮件 "; 

if($myrow["yanzheng"]==0) // 当 该 字段 值 为 0 时 为 系统 邮件 

echo "系统 邮件 :"; 

?> 


根据 获取 数据 库 中 相应 的 字段 内 容 判 断 邮件 类 型 后 ， 执 行 输出 操作 ， 当 判断 邮件 类 型 为 普通 邮件 
时 ， 输 出 “回复 ”按钮 ， 并 添加 隐藏 域 将 邮件 的 发 送 者 和 标题 等 信息 作为 隐藏 域 传递 到 写 信 操 作 页 面 ， 
当 判 断 邮件 的 类 型 为 好 友 验 证 邮件 时 ， 输 出 “加 为 好 友 ” 按 钮 ， 将 自己 的 名 称 和 发 送 请 求 者 的 会 员 名 
称 作为 隐藏 域 传递 给 add_friend.php 页 进行 处 理 ; 如 果 判 断 邮 件 类 型 为 系统 邮件 或 者 已 验证 邮件 时 ， 则 
没有 执行 选项 输出 。 程 序 相关 代码 如 下 : 


<?php 
if($myrow["yanzheng"]==1) { // 如 果 信件 为 好 友 验 证 信件 


@ 
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S$tb_mail_id=$_GET["mail id"]; 
qtb_ friend=$myrow["tb_mail_sender]; /发 送 请 求 者 
S$tb_my=$myrow["tb_receiving_person"]; // 被 请 求 认证 者 
echo "<form action=\"add friend.php\ method=\"post\" >"; 

echo "<input type=\"hidden\" name=\"tb_friend\" value=\"".$tb_friend."\" >"; 
echo "<input type=\"hidden\" name=\"tb_my\" value=\"".$tb_my."\" >"; 

echo "<input type=\"hidden\" name=\"tb_mail_id\" value=\"".$tb_mail_id."\" >"; 
echo "<input input class=\"inputsty\" type=\"submit\" value=\" 加 为 好 友 \">"; 
echo "</form>"; 


人 

if($myrow["yanzheng"]==2) { // 如 果 信件 为 普通 邮件 
S$receiver=$myrow["tb_mail_sender'"]; /转换 发 信者 作为 收 信人 
$sender=$myrow["tb_receiving_person"]; /转换 收 信者 转变 为 发 信人 
Sinclude=$myrow["tb_mail_content"]; // 信 件 内 容 


$subject=$myrow["tb_mail_subject"]; 

echo "<form action=\"send_mail.php?sender=".$sender."&&mails= 写 信 \" method=\"post\" >"; 
echo "<input type=\"hidden\" name=\"receiver\" value=\"".$receiver."\" >"; /添加 隐藏 域 
echo "<input type=\"hidden\" name=\"sender\" value=\"". $sender."\" >"; 

echo "<input type=\"hidden\" name=\"include\" value=\"".$include."\" >"; 

echo "<input type=\"hidden\" name=\"subject\" value=\"".$subject."\" >"; 

echo "<input class=\"inputsty\"type=\"submit\" value=\" 添 加 回复 \">"; 

echo "</form>"; 


} 


和 


当 判 断 邮件 为 验证 邮件 后 ， 单 击 “ 加 为 好 友 ” 按 钮 ， 将 当前 登录 的 会 员 名 称 和 发 送 请 求 的 会 员 名 
称 作为 传递 值 传递 到 add_friend.php 页 面 进行 处 理 , 添加 成 功 后 , 自动 向 申请 好 友 的 会 员 发 送 系 统 邮件 ， 
告 之 对 方 已 经 将 其 加 为 好 友 。 同 时 也 将 验证 信件 字段 更 改 为 “已 验证 ”状态 。 其 中 ， 添 加 好 友 处 理 页 
add_friend.php 的 程序 代码 如 下 : 


<?php 
include("conn/conn.php"); 
$tb_friend=$_POSTI["tb_friend"]; 
$tb_my=$_POST["tb_my"]; 
$tb_mail_id=$_POST["tb_mail id"]; 
S$tb_date=date("Y-m-d"); 
$querys=mysql_query("select * from tb_forum_user where tb_forum_user.tb_forum_user='$tb_friend™"); 
if(mysql_num_rows($querys)>0) { 
// 通 过 验证 后 字段 改 为 3 
$queryss=mysql_query("update tb_mail_box set yanzheng = 3 where tb_mail_id =$tb_mail_id"); 
// 向 发 送 请 求 的 用 户 发 送 系统 通知 

$tb_mail_subject=$tb_my." 已 经 通过 好 友 申 请 验证 "; 
$tb_mail_content=$tb_my." 已 经 通过 了 您 的 好 友 申请 !"; 

$querys=mysql_query("insert into tb_my_friend(tb_my,tb_friend,tb_date)values('$tb_my','$tb_friend','$tb_ 
date) 

$query=mysql_query("insert into tb_mail_box(tb_receiving_person,tb_mail_subject,tb_mail_content,tb_mail_ 
sender,tb_mail_date,yanzheng )values('$tb_friend','$tb_mail_subject','$tb_mail_content',' 系 统 信息 ',$tb_date',0)"); 

if($query==trueX{ 
echo "<script>alert(' 验 证 通过 ! 已 将 对 方 加 为 好 友 ');history.back();</script>"; 
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} 
Jelse{ 
echo "<script>alert( 对 不 起 ， 不 存在 该 用 户 !");history.back();</script>"; 


’ 
?> 


删除 指定 信息 的 操作 可 以 参考 7.2.8 节 中 的 内 容 ， 这 里 不 再 袭 述 。 

(3) 发 信 通 过 browse_send_mail.php 和 browse_send_mail_content.php 两 个 文件 来 完成 。 通 过 
browse_send_mail.php 文件 输出 数据 库 中 存储 的 发 送 记 录 ， 并且 根据 信息 的 标题 进行 分 页 输出 ， 设 置 超 
链接 ， 链 接 到 browse_send_mail_content.php 文件 ， 在 该 文件 中 输出 发 送信 息 的 详细 内 容 。 

由 于 篇 幅 所 限 ， 其 他 两 个 功能 的 程序 代码 这 里 没有 给 出 ， 完 整 代码 请 参看 本 书 附带 光盘 中 的 内 容 。 


7.7.3 我 的 好 友 管 理 
我 的 好 友 功 能 也 是 从 7.4.3 节 send_forum_content.php 文件 中 设置 的 超 链 接 开始 。“ 我 的 好 友 ” 链 


接 指向 my_friend.php 文件 ， 在 该 文件 中 完成 添加 好 友 的 操作 ， 并 且 向 好 友 发 送 一 条 验证 信息 。 其 运行 
结果 如 图 7.40 所 示 。 


A MINGRISOFT 实时 升级 服务 


询 福 甘 之 亲 共享 nD 西 万 双开 发 成 暴 . 妇 训 异 4 不 到 天 级 新 体 洽 


用 户 呢 蘑 2 14osE 个 人 信息 ”地 天 是 : 2011-09-14 


:区 于 

收 件 人: 区 家 注意 : 蛋 件 人 的 注册 刷 户 名 
主题: [于 耳 ， 允 家 

EE ET DE | 


ESE 
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图 7.40 我 的 好 友 模块 


在 my_friend.php 文件 中 , 创建 form 表单 ， 实 现 好 友 的 提交 , 将 数据 提交 到 my_friend_ok.php 文件 
中 进行 处 理 。 在 提交 到 处 理 页 my_friend_ok.php 后 ， 首 先 需要 判断 申请 的 好 友 对 象 是 否 为 自己 ， 如 果 
添加 对 象 是 自己 ， 则 给 出 警告 提示 。 然 后 通过 查询 语句 查询 双方 好 友 列 表 ， 判 断 申 请 者 和 被 申请 者 是 
否 至 少 有 一 方 已 存在 对 方 名 称 ， 如 果 至 少 有 一 方 存 在 对 方 名 称 ， 则 同样 不 能 执行 添加 操作 ， 并 返回 警 
告 提示 ; 如 果 双 方 好 友 列 表 中 都 不 存在 对 方 会 员 名 ， 则 执行 提交 ， 并 向 被 申请 者 发 送 一 条 验证 信息 。 
在 得 到 对 方 验 证 之 前 ， 申 请 者 不 能 重复 进行 好 友 添 加 操作 ， 直 到 对 方 验证 并 同意 添加 好 友 ， 则 完成 添 
加 好 友 操 作 过 程 。 其 中 ，my_friend_ok.php 文件 的 程序 代码 如 下 : 


<?php session_start(); include("conn/conn.php"); /连接 数据 库 
$tb_my=$_POST['my"]; // 获 取 表 单 中 提交 的 数据 
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S$tb_friend=$_POST["friend"]; // 获 取 表 单 中 提交 的 数据 
$tb_date=date("Y-m-d"); // 获 取 当 前 时 间 
S$tb_receiving_person=$_POST["receiving_person"]; /| 获取 接收 人 
S$tb_mail_subject=$_POST["mail_subject"]; /获取 信息 主题 
S$tb_mail_content=$_POST["mail_content"]; /获取 信息 内 容 
S$tb_mail_sender=$_POST["mail_sender"]; /获取 发 送 人 
S$tb_mail_date=date("Y-m-d"); 
S$tb_friend_tmp=$tb_my; /对 接收 人 和 发 送 人 重新 赋值 
S$tb_my_tmp=$tb_friend; 
if($tb_my==$tb_friend){ // 判 断 添加 的 对 象 是 否 为 自己 
echo "<script>alert(' 你 想 自己 加 自己 吗 ? ');history.back();</script>"; 
}else{ 
$querychkk=mysql_query("select tb_friend from tb_my_friend where tb_my="$tb_my_tmp"); 
$jcount=0; // 判 断 对 方 好 友 列表 中 是 否 存在 自己 名 称 


while($myrow=mysql_fetch_array($querychkk))f 
if($myrow[tb_friend]==$tb_my) 
$jcount++; 


} 
$kcount=0; // 判 断 自己 好 友 列 表 中 是 否 存 在 对 方 名 称 
$querychk=mysql_query("select tb_friend from tb_my_friend where tb_my='$tb_my"); 
while($myrow2=mysql_fetch_array($querychk))f 
if($myrow2['tb_friend']==$tb_friend) 


$kcount++; 
if($kcount>0) { 
if($jcount>0) // 如 果 双 方 好 友 列 表 中 存在 对 方 名 称 
echo "<script>alert(' 你 们 已 经 是 好 友 了 ， 不 必 再 进行 添加 了 ');history.back();</script>"; 
else if($jcount==0) // 如 果 对 方 好 友 列 表 不 存在 自己 名 称 
echo "<script>alert(' 好 友 验 证 已 发 送 ， 请 耐心 等 待 对 方 确认 ');history.back();</script>"; 
Jelse{ // 如 果 双 方 好 友 列 表 中 都 不 存在 对 方 名 称 ， 则 进行 添加 好 友 操 作 


$querys=mysql_query("select * from tb_forum_user where tb_forum_user='$tb_receiving_person'"); 
if(mysql_num_rows($querys)>0X{ 

$querys=mysql_query("insert into tb_my _friend(tb_my,tb_friend,tb_date)values('$tb_my','$tb_friend', 
'$tb_date')"); 

$query=mysql_query("insert into tb_mail_box(tb_receiving_person,tb_mail_subject,tb_mail_content, 
tb_mail_sender,tb_mail_date,yanzheng )values('$tb_receiving_person','$tb_mail_subject','$tb_mail_content','$t 
b_mail_sender','$tb_mail_date’,1)"); 

if($query==true}{ 
echo "<script>alert( 短信 息 发 送 成 功 !);history.back();</script>"; 


} 
}else{ 
echo "<script>alert(' 对 不 起 ， 不 存在 该 用 户 !");history.back();</script>"; 
} 
} 
} 


?> 


在 完成 好 友 的 添加 之 后 ， 在 会 员 登 录 成 功 的 页 面 中 有 一 个 “我 的 好 友 ” 超 链接 ， 单 击 该 链接 进入 
browse_friend.php 文件 中 ， 在 该 文件 中 可 以 查看 到 所 有 的 好 友 。 当 单 击 好 友 的 名 称 时 将 链接 到 person_ 


_ 国 
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data.php 文件 ,通过 该 文件 可 以 查看 到 好 友 的 详细 信息 .在 browse_friend.php 文件 中 ,还 创建 了 一 个 form 
表单 ， 将 数据 提交 到 delete_friend.php 文件 中 ， 实 现 对 指定 好 友 进 行 删除 的 操作 。 删 除 过 程 同样 可 以 参 
考 7.2.8 节 中 的 内 容 ， 虽 然 整体 思路 相同 ， 但 是 代码 有 细微 变动 。 其 运行 结果 如 图 7.41 所 示 。 


认识 时 间 


2008-0(-18 
分 页 : 
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7.8 后 台 管 理 


7.8.1 后 台 管 理 概述 


论坛 后 人 台 管 理 作为 整个 论坛 模块 中 不 可 缺少 的 因素 之 一 ， 其 不 但 可 以 实现 各 种 帖子 的 删除 、 置 项 、 
结 帖 等 管理 操作 ， 而 且 可 以 对 论坛 会 员 的 账号 执行 权限 设置 、 账 号 删除 等 操作 。 其 中 ， 后 台 管理 主要 
包括 会 员 管理 、 回 帖 管理 、 顶 帖 管 理 等 多 个 管理 模块 ， 其 运行 结果 如 图 7.42 所 示 。 
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图 7.42 ”后 台 管理 功能 展示 
7.8.2 后台 登录 


和 前 台 登 录 一 样 ， 管 理 员 要 想 对 论坛 进行 管理 ， 首 先 必须 执行 登录 操作 ， 在 管理 员 登 录 成 功 后 ， 
页 面 才能 跳 转 到 论坛 管理 后 台 主 页 .管理 员 与 普通 用 户 的 身份 判别 是 根据 数据 表 tb_forum_user 中 的 tb_ 
forum_type 字段 来 决定 的 ， 当 该 字段 值 为 1 时 ， 用 户 身份 为 普通 会 员 ， 当 其 值 为 2 时 ， 表 明 该 用 户 身 
份 为 管理 员 。 
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后 台 管 理 员 登录 由 enter_ manage.php 与 enter_manage_ok.php 两 个 文件 组 成 。 
(Cl)enter manage.php 用 于 显示 后 台 登 录 界 面 , 该 文件 由 两 部 分 组 成 , 第 一 部 分 为 JavaScript 内 容 ， 
用 于 判断 输入 的 管理 员 账 号 、 密 码 以 及 验证 码 是 否 为 空 ， 其 关键 代码 如 下 : 


<script language="JavaScript" type="text/javascript"> 
function check_user(form){ 


ifform.tb_uservalue=="){ 1/ 判断 用 户 名 是 否 为 空 
alert(" 请 输入 管理 名 "); 
form.tb_user.select(); 
retum(false); 
上 
if(form.tb_pass.value=="){ 1/ 判断 输入 的 密码 是 否 为 空 


alert(" 请 输入 登录 密码 ! "); 

form.tb_pass.select(); 

retum(false); 

} 

if(form.tb_validate.value==""){ 1/ 判断 输入 的 验证 码 是 否 为 空 

alert(" 请 输入 验证 码 !"); 

form.tb_validate.select(); 

return(false); 

} 

return(true); 


} 
</script> 
第 二 部 分 为 表单 设计 部 分 ， 用 于 获取 管理 员 登 录 的 用 户 名 、 密 码 、 验 证 码 等 登录 信息 ， 其 关键 代 
人 码 如 下 : 


<table align="center width="750" height="24" border="0" cellpadding="0" cellspacing="0" background="images/ 
bg_13(1).JPG"> 
<form action="enter_manage_ok.php" method="post" name="form1" id="form1" onSubmit="return check_user 
(this)"> 
<tr> 
<td width="71" align="right"><span class="STYLE1"> 用 户 名 : </span></td> 
<td width="138"><input type="text" name="tb_user" size="18" /></td> 
<td width="68" align="right" class="STYLE1"> 密 码 : </td> 
<td width="145"><input type="password" name="tb_pass" size="18" /></td> 
<td width="80" align="right" class="STYLE1"> 验 证 码 : </td> 
<td width="94"><input type="text" name="tb_validate" size="10" /></td> 
<td width="50" align="center"><img src="../tb_validate.php"></td> 
<td width="87" align="center"><input type="submit" name="Submit" value=" 登 录 "></td> 
<td width="17">&nbsp;</td> 
</tr> 
</form> 
</table> 


(2) enter_manage_ok.php 封装 一 个 check_user 类 ， 定 义 check_input0 方 法 对 表单 中 提交 的 用 户 名 


和 密码 进行 验证 。 关 键 代码 如 下 : 
@ 
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<?php session_start(); 
$tb_validate=trim($_POST[tb_validate]); 
class check_user{ 


var $tb_user; // 定 义 用 户 名 变量 
var $tb_pass; /定义 用 户 密码 变量 
var $tb_validate; /定义 验证 码 变量 
function check_user($x,$y,$mj{ /创建 构造 函数 


S$this->tb_user=$x; 
S$this->tb_pass=$y; 
S$this->tb_validate=$m; 


} 
function check_input(X{ 
if(strval($this->tb_validate)!=$_SESSION["validate1"]){ /| 判断 用 户 输入 的 验证 码 是 否 正确 
echo "<script>alert(' 验 证 码 输入 错误 !");history.go(-1);</script>"; 
exit; 
人 /加 载 数据 库 相 关 配 置 
$sql=mysql_query("select tb forum_user from tb_forum_user where tb _forum_type=2 and tb_ forum_ 
User=".$this->tb_user."",$conn); // 执 行 数据 库 查 询 
$info=mysql_fetch_array($sql); 
if($info==false){ // 判 断 用 户 名 是 否 存 在 
echo "<script>alert(' 对 不 起 ， 不 存在 该 用 户 !");history.back();</script>"; 
exit; 
jelse{ 


$sql=mysql_query("select tb_forum_user from tb_forum_user where tb_forum_type=2 and tb_forum_ 
User=".$this->tb_user." and tb_forum_pass=".$this->tb_pass."",$conn); 
$info=mysql_fetch_array($sql); 


if($info==false){ // 判 断 输 入 的 密码 是 否 正确 
echo "<script>alert( 对 不 起 ， 密 码 输 入 错误 !");history.back();</script>"; 
exit; 
Jelse{ 
if($_SESSION["admin_user"]!=") / 淹 断 Session 变量 是 否 为 空 
session_unregister("admin_user"); /销毁 Session 变量 
} 
$_SESSION["admin_user"]=$this->tb_user; /重新 注册 Session 变量 
echo "<script>alert(' 登 录 成 功 !");window.location.href='index.php';</script>"; 
} 
} 
mysql_close($conn); // 关 闭 数据 库 连接 
} 
} 
$chk=new check_user($_POST[tb_user],md5($_POSTItb_pass]),$tb_validate); 。 // 初 始 化 该 类 
$chk->check_input(); // 调 用 类 的 函数 
?> 


7.8.3 ”后 台 管理 主页 设计 


后 台 管理 主页 的 页 面 设计 其 实 没 有 太 多 实质 性 的 技术 可 言 ， 其 主要 是 为 不 同 的 管理 栏目 设置 超 链 


@ 
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接 ， 然 后 应 用 switch 语句 根据 超 链接 传递 的 参数 值 进行 判断 ， 进 而 完成 不 同 栏目 的 加 载 功能 。 
在 index.php 文件 中 ， 首 先 判断 当前 访问 管理 页 的 用 户 权限 。 关 键 代 码 如 下 : 


<?php 
session_start(); /开启 Session 设置 
include("../conn/conn.php"); /1 加载 数据 库 配置 文件 
if ($page==") { // 判 断 当 前 页 码 是 否 为 空 
$page=1; 
if$_SESSION["admin_user]=="")f /| 判断 当前 用 户 权限 
echo "<script>alert(' 禁 止 非法 登录 !");window.location.href='enter_manage.php';</script>"; 
exit; 
}else{ 
?> 


然后 为 不 同 的 栏目 创建 超 链接 ， 并 且 定 义 超 链接 的 参数 值 。 关 键 代码 如 下 : 


<table width="100%" height="275" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td height="80" colspan="3"><img src="images/index_2.jpg" width="1003" height="80"></td> 
</tr> 
<tr> 
<td height="24" colspan="3" ><img src="../images/index_4.jpg" width="1003"></td> 
</tr> 
<tr> 
<td width="153" valign="top" background="images/index_5.jpg"><table width="150" height="193" border= 
"0" cellpadding="0" cellspacing="0" background="images/index_5.jpg"> 
<tr> 
<td height="24" align="center"><a href="index.php?title= 会 员 管理 ”class="STYLE3"> 会 员 管 理 </a> 


</td> 
</tr> 
<tr> 
<td height="24" align="center"><a href="index.php?title= 公 告 管理 ”class="STYLE3"> 公 告 管理 </a> 
</td> 
</tr> 
<tr> 
<td height="25" align="center"><a href="index.php?title= 帖 子 类 别管 理 ”class="STYLE3"> 帖 子 类 别管 
理 </a></td> 
</tr> 
<!-- 省 略 部 分 代码 一 > 
</table></td> 
<td width="7" height="400" bgcolor="#EFF3F7">&nbsp;</td> 
<td width="1075" align="left" valign="top"> 


最 后 应 用 switchO 语 句 ， 根 据 URL 地 址 中 的 title 参数 进行 判断 ， 加 载 不 同 的 栏目 页 面 ， 并 将 加 载 
的 内 容 输 出 到 指定 的 层 中 ， 其 关键 代码 如 下 : 


<td width="1075" align="left" valign="top"> 
<div style=" width:840px;"> 


@ 
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<?php 
if(lempty($_GET['title"]) /获取 相关 URL 参数 
$title=""; 
else 
Stitle=$_GETI"title"]; 
switch($titleX{ /通过 GET 方法 获取 URL 中 参数 title 的 值 
case "会 员 管理 ": 
include("leaguer_admin.php"); /加 载 相应 的 页 面 
break; 
case "公告 管理 ": 
include("send_affiche.php"); 


break; 
case "帖子 类 别管 理 ": 
include("append_small_type.php"); 
break; 
/省 略 部 分 代码 

case ™: 
include("send_affiche.php"); 
break; 

3 

?> 

</div> 

</td> 


7.9 本 章 小 结 


本 章 运 用 软件 工程 的 设计 思想 ， 通 过 一 个 完整 的 论坛 来 讲解 一 个 系统 的 开发 流程 。 同 时 ， 在 论坛 
的 开发 过 程 中 ， 讲 解 了 如 何 屏 蔽 回帖 ， 以 及 帖子 的 发 布 、 回 复 、 收 藏 等 功能 ， 使 整个 系统 的 设计 思路 
更 加 清晰 。 通 过 本 章 的 学 习 ， 读 者 不 仅 可 以 了 解 一 般 网 站 的 开发 流程 ， 而 且 可 以 对 论坛 中 的 内 容 进 行 
回帖 、 发 表 帖子 等 。 


AS 立 
性 


在 线 支付 模块 


(PHP+Smartyt+MySQL 实现 ) 


随 着 Internet 的 发 展 , 电子 商务 已 经 成 为 21 世 纪 网 络 发 展 的 主流 ， 
网 上 购物 和 网 上 支付 已 经 成 为 一 种 时 尚 。 目 前 ， 国 内 企业 和 个 人 都 纷 
纷 加 入 到 阿里 巴巴 等 一 系列 大 型 电子 商务 网 站 中 ,通过 网 络 进行 B2B、 
B2C 和 C2C 的 网 上 交易 ， 在 进行 网 上 交易 的 同时 ,付款 方法 也 在 随 着 
网 络 的 发 展 发 生 着 变化 ， 在 线 支付 方式 的 快捷 和 方便 已 经 得 到 越 来 越 
多 人 的 认可 。 为 了 让 广大 PHP 爱好 者 能 开发 出 功能 更 加 完善 的 电子 商 
务 类 网 站 ， 在 本 模块 中 ， 将 介绍 如 何 通过 支付 宝 完 成 电子 商务 网 站 在 
线 支付 操作 ， 以 及 订单 生成 和 订单 打印 技术 的 实现 方法 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 


ml 


各 于 于 于 于 于 至 


动态 生成 订单 号 

订单 打印 

支付 宝 支 付 

订单 查询 

统计 购物 车 中 商品 金额 

向 购物 车 中 添加 商品 

清空 购物 车 

Smarty 模板 实现 网 页 动静 分 离 


》 PHP 典型 模块 开发 全 程 实录 


8.1 在 线 支 付 模块 概述 


8.1.1 功能 概述 


在 线 支付 与 购物 车 相同 ， 都 是 电子 商务 网 站 必 不 可 少 的 一 部 分 。 本 模块 在 购物 车 模块 的 基础 上 ， 
将 填写 收 货 人 信息 、 生 成 订单 、 订 单打 印 、 订 单 预览 和 订单 查询 作为 辅助 ， 最 终 完 成 支付 宝 在 线 支付 
的 操作 。 在 线 支 付 模块 的 功能 结构 如 图 8.1 所 示 。 


存续 支付 模块 
I 


填 | | | 风 | 连 
采 | 屋 | | 亲属 共 | | 拓 | 必 | 度 
人 | | 展 | | 这 | 省 | 自 二 订 | 歧 | 歧 
人 | | 示 览 说 | 单间 | 插 
更 商 | 出 
导购 | 区 | 恒 | 9| | 继 | 品 | 避 
量 | | 买 | 示 | 庭 | 除 | | 续 | 会 | 性 
罗 | | 高 | 避 | 刚 | 说 | | 多 | 司 | 属 
刘 咏 | 数 | 网 | 避 | 后 | | 及 
里 计 | | 计 


图 8.1 在 线 支 付 模块 的 功能 结构 图 


8.1.2 在 线 支付 操作 流程 


在 线 支 付 模块 的 操作 流程 与 购物 车 十 分 相似 , 唯一 区 别 就 是 增加 了 支付 宝 支付 和 订单 查询 的 功能 。 
其 流程 如 图 8.2 所 示 。 


改 购 物 车 | 
中 商品 数量 
订单 查询 | 
中 指定 商品 
已。 | -HR 
清空 购物 车 一 | 一 | 人 信息 
用 户 查看 购物 车 内 商品 继续 购物 。 | | 
站 和 
商品 金额 总 计 “| 吕 
订单 打印 
支付 宝 支 付 | 一 < 一 | 提交 订单 |< 
订单 预览 


8.2 在 线 支付 的 操作 流程 
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8. 证 3 程序 预览 


在 线 支付 模块 是 电子 商务 网 站 收银 台中 的 一 个 子 功能 ， 为 了 让 读者 对 本 模块 有 个 初步 的 了 解 和 认 
识 ， 下 面 列 出 几 个 典型 功能 的 页 面 。 

明日 购物 商城 网 站 填写 收 货 人 信息 的 页 面 如 图 8.3 所 示 ， 该 页 面 完成 收 货 人 姓名 、 电 话 、 联 系 地 址 
等 信息 的 添加 操作 。 


订单 信息 
收 货 人 ， 证 村 邮编 ， [130025 
联系 电话 ， |1360433™ 


地 址 。 及 春 ] 
送 货 方 式 ， 平邮 ~ 
付款 方式 ， 支 人 宝 ~ 
提交 订单 
后 @ internet | 保护 模式 要 用 往 - 搞 100% > 
= 于 


图 83 收 货 人 信息 添加 页 面 


收 货 人 信息 填写 完毕 之 后 ， 就 是 提交 订单 ， 并 且 生 成 订单 号 ， 进 而 完成 订单 的 预览 、 打 印 以 及 在 
线 支付 等 操作 。 生 成 订单 信息 的 页 面 如 图 8.4 所 示 。 


| Bhp/ 8 有 681.591yfd=1306200330 - 林 昌 提交 成 9 - Windows Intemet Explorer [EX 一 
订单 查看 
订单 号 ， 1306200330 订单 时 间 : 2011-05-24 092530 
下 单 人 : mrsoft 收 货 人 ， 湛 殉 
邮编 : 130025 电话 : 13604338784 
地 址 :长春 市 
送 货 方式 ， 平邮 付款 方式 : 银行 转账 
订单 内 容 
商品 名 数量 价格 价格 折 率 合计 
数码 相机 1 16992 元 9 折 16992 元 
家 庭 影院 1 43992 元 9 折 43992 元 
总 消费 : 6098.4 元 
请 您 ,在 一 周 A 扩 的 交付 方式 人 和 和 的 亲生 后 请 有 时 通知 牧 们 。 
注意 : 请 记 住 订单 号 。 以 便 日 后 查询 及 有 疑 jeBj 使 用 
打印 岳 览 。 我 要 打印 ”支付宝 支付 


8.4 生成 订单 信息 页 面 


在 图 8.4 中 单 击 “ 打 印 预览 ”按钮 ， 将 进入 打印 预览 页 面 ， 其 运行 效果 如 图 8.5 所 示 。 
单 击 “ 我 要 打印 ”按钮 ， 将 执行 订单 的 打印 操作 ， 其 运行 效果 如 图 8.6 所 示 。 
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EE 
屿 | 加 -四 | 入 | 晶 | 加 -图 1 -| 68EX i120 


订单 查看 
可 第 时 间 : 2011-05-24093347 


藉 吉 多 ! 订单 拔 交 式 功 。 
请 人 在 一 周 内 想 人 的 文 付 入 区 进行 汇款 汇款 时 法 强 伯 的 订 亲 要 | 汇款 后 潭 及 时 于 知 我 们 。 打印 到 文件 ， 
这 芝 半生 届 入 让 3 天 是 rp Ee ] 


全 MG 1 放 
口 自动 分 太 人 @ 
.22 


EED GW LERw | 


图 8.5 订单 打印 预览 页 面 图 8.6 订单 打印 
8.2 数据库 设计 
本 模块 与 购物 车 模块 同属 于 电子 商务 网 站 中 一 个 子 功能 ， 其 数据 库 的 使 用 是 统一 的 ， 都 是 MySQL 
数据 库 。 与 购物 车 模块 存在 区 别 的 地 方 就 是 调用 的 是 不 同 的 数据 表 。 
8.2.1 创建 数据 库 
再 一 次 看 一 下 电子 商务 网 站 所 使 用 的 数据 表情 况 ， 如 图 8.7 所 示 。 


胃 服务 器 : localhost ， 忆 数据 库 : db_business 
而 某 攀 “部 SQL 万 搜索 局 查 词 稳 SS 出 本 Import 优 扣 作 约 权 限 加 出 芭 
表 报 作 记录 教 四 。 类 型 整理 


讽 明 
了 admin 加 区 国有 面 XxX 2 MWSAM uti8_unicode_c ”管理 员 表 
也 cass 加 上 攻 国 线 硬 Xx 但 MYSAM 。 uti8_unicode_c 商品 类 别 表 
也 commo 园 办 因 钴 国 X 4 MSAM 。 uli8_unicode_c 。 商品 信息 表 
也 fm 国 办 国庆 看 X 人 7 MYSAM utf8_unicode_c 。 商品 订单 表 
tmks 图 攻 国 线 面 XX 2 MWSAM ut8_unicode_c 。 友情 镑 接 表 
了 bopmion 与 上 你 定 X 0 MMSAM ulB_unicode_c 商品 留言 表 
也 public 国 具 国 半 硬 X 4 MSAM uli8_unicode_d 商品 公告 表 
user 转 具 加 广 国 X 5 MMSAM ui8_unicode_ci 。 会 员 信 息 夫 
8 个 表 总 计 48 WYISAM utf8_general_ci 


8.7 ”电子 商务 网 站 数据 表 
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8.2.2 创建 数据 表 


本 节 介 绍 与 在 线 支 付 有 直接 关系 的 数据 表 的 结构 。 

tb_commo《〈 商 品 信息 表 ) 

商品 信息 表 主 要 用 于 存储 关于 商品 的 相关 信息 ， 其 结构 如 图 8.8 所 示 。 

轩 服务 器 : localhost ”局 数据 库 : db_business 》 园 表 : tb_commo 
字段 类 型 整理 属性 Nul 。 黑 认 格外 说明 
否 auto_increment 。。 自动 闹 号 

商品 名 称 

picsmuljpg 商品 图 片 
商品 介绍 
添加 时 间 
商品 产地 
商品 型 号 
商品 类 型 
商品 名 称 

| 商品 库存 

0 销售 量 


到 友 到 到 到 巴 到 到 到 到 到 叫 叫 叫 叫 


图 8.8 商品 信息 表 结 构 


tb_form (订单 信息 表 ) 
订单 信息 表 主 要 用 于 存储 关于 订单 的 相关 信息 ， 其 结构 如 图 8.9 所 示 。 


加 服务 器 :localhost 》 数据 库 : db_business ， 四 表 :tb_form 
类 型 整理 属性 默认 


Man 说 明 
理 auto_increment 。 自动 谢 号 
UW8_unicode_d 否 订单 号 
we_unicode_dl 否 商品 ID 
ut8_unicode_ci 理 商品 名 称 
UNB_unicode 否 商品 数量 
ta_unlcode dl EE 商品 单价 
ut8_unicode_ci 理 打折 率 
Wa_unicoge_ oi 否 消费 台灯 
we_unicode_dl 否 消费 者 
WB_unicode_ di 否 收 顷 人 姓名 
utiB_unicode_oi 否 收 贷 地 址 
ut8_unicode_ci 理 联系 电话 
we_unicode_ci 否 邮编 
we_unicode_ci 否 付 基 方 式 
ut8_unicode_Gi 理 邮寄 方式 
否 。 CURRENT_TIMESTAMP 订单 日 其 
否 订单 枯 态 


图 8.9 订单 信息 表 结构 


此 外 ， 还 有 管理 员 表 、 商 品 公告 表 、 用 户 信息 表 、 友 情 链接 表 和 商品 留言 表 ， 限 于 篇 幅 ， 这 里 不 
再 介绍 ， 读 者 可 参见 本 书 附 赠 光盘 中 的 数据 库 文件 。 
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8.3.1 会 员 信 息 模块 概述 


用 户 登录 后 ， 即 可 看 到 会 员 信息 模块 。 
在 这 里 ， 可 以 进行 查看 或 修改 个 人 信息 及 密 


:2 


码 、 查 看 购物 车 和 安全 退出 等 操作 。 本 节 只 下 
会 员 信息 模块 中 的 “会 员 中 心 ”和 “安全 sa: 


退出 ”进行 讲解 。 会 员 信 息 模块 的 运行 效果 


祖 各 总额 : 0 
如 图 8.10 所 示 。 
真实 姓名 : 潘 全 
会 员 E 身份 证 号 : 220322821103653 
32 i i 中 本 移动 电话 ; 136043386784 
固定 电话 : 0431-84972266 
当 单 击 “ 会 员 中 心 ” 超 链接 时 ， 会 回 传 给 El pp@sina.com 
当前 页 一 个 page 值 ， 当 前 页 根据 这 个 page 值 CD 
来 载 入 member.php 文件 。 HE 
CE CE 


1. 创建 PHP 页 面 
与 登录 模块 设计 不 同 ， 本 节 首 先 来 创建 


2. 创建 模板 页 


图 8.10 会 员 信息 模块 的 运行 效果 


PHP 页 面 。 因 为 该 模块 中 的 模板 需要 使 用 数据 库 中 的 数据 及 一 些 动 态 信息 ， 这 些 都 需要 在 PHP 页 中 先 
行 获取 及 处 理 ， 然 后 再 传 给 模板 页 。 会 员 中 心 页 面 的 代码 请 参考 8.6 节 技 术 提炼 分 析 中 的 内 容 。 


该 模块 包括 查看 信息 模板 及 修改 密码 模板 ， 都 存储 于 member.tpl 模板 文件 中 。 关 键 代 码 如 下 : 


<link rel="stylesheet" href="css/member.css" /> 


<script language="javascript" src="js/member.js"></script> 


{if $check=="find" } 


<p align="left">{$smarty.session.member}&gt;&gt;&gt;<a href='?page=hyzx' id="mem"> 查 看 信息 </a>&gt;&gt; 
<a href='?page=hyzx&action=modify id="mem"> 修 改 密码 </a></p> 
<table id="member" width="300" border="0" cellpadding="0" cellspacing="0"> 
<form id="member"” name="member" method="post"” action="modify_ pwd_chk.php” onSubmit="retum pwd 


(member)"> 
<tr> 


<td height="25" colspan="2" align="center" valign="middle" id="first"><font color="#f0f0f0"> 修 改 密码 </font></td> 


</tr> 
.….// 省 略 了 部 分 代码 


<tr> 


<td height="30" colspan="2" align="center" valign="middle"> 
<input id="enter" name="enter" type="submit" value=" 修 改 " /></td> 


</tr> 


他 
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</form> 
</table> 
{else} 
<p align="left">{$smarty.session.member}&gt;&gt;&gt;<a href='?page=hyzx' id="mem"> 查 看 信息 </a>&gt; 
<a href='?page=hyzx&action=modify id="mem"> 修 改 密码 </a></p> 
{section name=pwd_id loop=$pwdarr} 
<table id='member width="500" border="0" cellpadding="0" cellspacing="0"> 
<form id="member name="member”method="post”action="modify_info_chk.php”onSubmit="retum mem 
(member)"> 
<tr> 
<td height="25" colspan="2" align="center" valign="middle" id="first"> 
<font color="#f0f0f0">{$pwdarr[fpwd_idj.name} 信 息 〈 不 可 更 改 信息 ) </font></td> 
</tr> 
<tr> 
<td width="25%" height="25" align="right" valign="middle" id="left"> 会 员 编号 :</td> 
<td height="25" align="left" valign="middle" id="right">&nbsp;{$pwdarr[pwd_id].id}</td> 
</tr> 
…// 省 略 了 部 分 代码 
<tr> 
<td height="30" colspan="2" align="center" valign="middle"> 
<input name="enter" type="submit" id="enter" value=" 修 改 " />&nbsp;&nbsp;&nbsp;&nbsp; 
<input name="reset" type="reset" id="reset" value=" 重 置 " /></td> 
</tr> 
</form> 
</table> 
{/section} 
{i 座 


3. 创建 脚本 文件 


该 模块 的 脚本 文件 和 用 户 注册 模块 类 似 ， 都 是 对 信息 的 合法 性 进行 验证 ， 如 信息 是 否 为 室 、 是 否 
符合 规范 等 ， 其 存储 于 08Wswmemberjs 中 ， 这 里 不 再 效 述 。 


4. 创建 处 理 页 
当 信息 验证 通过 后 , 系统 将 跳 转 到 处 理 页 modify_info_chk.php 进行 信息 处 理 。 信息 处 理 页 代码 如 下 : 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文 件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 

$sql = "update tb_user set realname=".$_POST[realname'].",card=".$_POST[Tcard].",tel=".$_POST[tel]."， 
phone=".$_POST[phone].",Email=".$_POST[email].",QQ='".$_POST['qq].",code=".$_POST[code']."， 
address=".$_POST['address]." where id = ".$_POST[userid].""; 

$arr = $admindb->ExecSQL($sql,$conn); 

if($arr) 

echo "<script>alert(' 修 改 成 功 ");location=('index.php');</script>"; 

else 

echo "<script>alert(' 修 改 失 败 ');history.go(-1);</script>"; 

?> 
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8.3.3 安全 退出 
当 用 户 需要 离开 网 站 时 ， 可 以 单 击 “ 安 全 退出 ” 超 链接 来 调用 logout0 函 数 ， 当 用 户 确认 退出 后 ， 
则 跳 转 到 logout 页 面 ， 销 毁 Session 并 回 到 首页 。 安 全 退出 所 涉及 的 页 面 及 代码 如 下 : 


代码 位 置 : 光盘 mm08Vs\info.js 
function logout(){ 


if(confirm(" 确 定 要 退出 登录 吗 ? ")X{ /输出 选择 框 ， 用 户 可 以 单 击 “ 确 认 ” 或 “取消 ”按钮 
window.open('logout.php','_parent',",false); /| 如果 用 户 确认 退出 ， 则 打开 logout.php 页 
}else 
retum false; 
} 
代码 位 置 : 光 税 \mr\08\logout.php 
<?php 


session_start(); 

header ("Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
session_destroy(); 

echo '<script>alert(\' 用 户 已 安全 退出 N);location=(\index.php\);</script>"; 

Ee 


8.4 收银 台 模 块 设计 


8.4.1 收银 台 模 块 概述 


当 用 户 停止 浏览 商品 准备 结账 时 ， 可 以 单 击 购物 车 页 面 中 的 “去 收银 台 ” 按 钮 ， 将 触发 onclick 事 
件 ， 调 用 formset(O 函 数 显示 订单 页 面 ， 当 用 户 提交 订单 后 ， 系 统 将 订单 保存 到 数据 表 tb_form 中 ， 同 时 
清空 购物 车 ， 并 显示 订单 信息 ， 提 醒 用 户 记 录 订 单 号 。 当 货款 发 出 后 ， 还 可 以 对 订单 进行 查询 。 收 银 
台 页 面 的 运行 结果 如 图 8.11 所 示 。 


税收 殷 台 - windows Internet Explorer [ET 
ED http/192.168.1.58/SOFT/05/sette. 喇 | 
订单 信息 
LE [130025 
aiid Tar 
地 址 ，| 攻 春 市 
详 线 记 式 ， 快递 。 一 
人 方式 ， 支付 宇 ~ 
提单 
| 
| 
越 ET ET 


8.11 收银 台 页 面 的 运行 结果 
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本 节 所 涉及 的 页 面 有 显示 订单 (formsetO 函 数 )、 填 写 订 单 (settle.php、settle.tp1)、 提 交 订 单 (settle_ 
chk.php) 、 反 馈 订 单 〈forminfo.php、forminfo.tpl1) 和 查询 订单 5 部 分 。 


8.4.2 显示 订单 


订单 信息 提交 页 面 的 输出 由 formset(O) 函 数 决定 ， 它 将 商品 信息 进行 整理 ， 通 过 open() 方 法 打开 
settle.php 页 来 显示 订单 ， 并 将 整理 后 的 商品 信息 传递 到 settle.php 文件 中 。formset0 函 数 存 储 于 
js\shopcar.js 文件 中 ， 其 关键 代码 如 下 : 


function formset(formX{ 

var uid = form.uid.value; 

varn_pre ='cnum'; /| 数量 
var lang = form.chk.length; 

if(lang == undefined){ 


var fst = form.chk.value; // 商 品 id 
var snd = form.cnum0.value; 1/ 购买 数量 
}else{ 


var fst= new Array(); 

var snd = new Array(); 

for(var i = 0; i < lang; i++){ 

var nm = n_pre+i.toString(); 

var stmp = document.getElementByld(nm).value; 
if(stmp == "||isNaN(stmp)X{ 

alert(' 不 允许 为 空 ， 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

snd[li] = stmp; 

var ftmp = form.chk[il.value; 
fst[i] = ftmp; 

} 

} 


open(settle.php?uid='+uid+'&fst='+fst+'&snd='+snd，blank,width=500 height=450' ,false); 


4 
四 因为 open() 方 法 使 用 了 _blank 参数 来 打开 一 个 新 的 页 面 ，Session 值 传 不 过 去 ， 所 以 这 里 
使 用 隐藏 域 来 传递 用 户 名 称 。 


8.4.3 填写 订单 


settle.php 直接 将 接收 的 值 传 给 settle.tpl 模板 ， 并 载 入 settle.tpl 模板 。settle.php 页 面 的 代码 如 下 : 


<?php 
session_start(); 
header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 


符合 


的 变 
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require("system/system.inc.php"); // 包 含 配 置 文件 
S$fst = $_GET[fst]; 

$snd = $_GET[snd'; 

S$uid = $_GET[uid']; 

$smarty->assign('"title',' 收 银 台 '); 

$smarty->assign('fst', $fst); 

$smarty->assign('snd', $snd); 

$smarty->assign('uid',$uid); 

$smarty->display('settle .tpl"); 

?> 


在 settle.tpl 模板 页 首先 载 入 settle.js 脚本 文件 , 调用 其 中 的 chkreginfo0 方 法 验证 表单 提交 的 值 是 否 


要 求 。 然 后 创建 表单 form_reg， 提 交 收 货 人 、 联 系 电话 、 联 系 地 址 等 信息 ， 同 时 还 将 PHP 页 传递 
量 以 隐藏 域 的 形式 提交 到 处 理 页 settle_chk.php。settle.tpl 模板 页 的 关键 代码 如 下 : 


<form name="form_reg" method="post" action="settle_chk.php" onSubmit="retum chkreginfo(form_reg,'all)"> 


8.4 


<tr> 

<td height="25" align="right" valign="middle" class="left"> 收 货 人 : </td> 

<td width="221" height="25" align="left" valign="middle" class="center">&nbsp; 
<input type="text" name="taker" size="20" onBlur="chkreginfo(form_reg,0)"> 
<div id="chknew_taker" style="color:#FF0000"></div> 

</td> 

<td width="56" align="right" valign="middle" class="center"> 邮 编 : </td> 

<td width="207" align="left" valign="middle" class="right">&nbsp; 

<input type="text" name="code" size="20" onBlur="chkreginfo(form_reg,1)"> 
<div id="chknew_code" style="color:#FF0000"></div></td> 

</tr> 

<!-- 省 略 了 部 分 代码 --> 

<tr> 

<td width="116" height="25" align="right" valign="middle" class="left"> 付款 方式 :</td> 
<td height="25" colspan="3" align="left" valign="middle" class="right">&nbsp;<select id="pay" name="pay"> 
<option value=" 银 行 转账 "> 银行 转账 </option> 

<option value=" 邮 局 汇款 "> 邮局 汇款 </option> 

<option value=" 支 付 宝 "> 支付 宝 </option> 

</select> </td> 

</tr> 

<tr> 

<td height="30" colspan="4" align="center valign="middle"> 

<input id="enter" name="enter" type="submit" value=" 提 交 订 单 " class="btn" /> 
<input id="fst" name="fst" type="hidden" value="{$fst}" /> 

<input id="snd" name="snd" type="hidden" value="{$snd}" /> 

<input id="uid" name="uid" type="hidden" value="{$uid}" ></td> 


</tr> 
</form> 
.4 ”处 理 订单 


处 理 页 settle_chk.php 获取 表单 中 提交 的 数据 , 根据 用 户 提交 的 商品 信息 ， 


新 查找 数据 表 tb_commo， 


他 
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并 从 数据 表 中 提取 商品 信息 ， 保 存 到 数组 中 ， 然 后 处 理 页 将 数组 作为 一 条 记录 添加 到 tb_form 表 。 
数据 添加 成 功 的 同时 , 处 理 页 会 根据 uid 找到 该 用 户 , 将 shopping 字段 清空 , 最 后 调用 forminfo.php 
页 来 显示 新 添加 的 订单 信息 。settle_chk.php 页 的 代码 如 下 : 


<?php 
header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 


$sql="insert into tb_form(formid,commo_id,commo_name,commo_num,agoprice,fold,total,vendee,taker,address, 
tel,code,pay_method, del_method,formtime,state)values("; 

$formid=time(); 

$tmpid = explode(,,$_POSTIYst]); 

$tmpnm = explode(,,$_POST[snd]); 

$number = count($tmpid); 

$tmpna = array(); 

$tmpvp = array(); 

$tmpfd = array(); 

S$tmptt = 0; 

if($number >1X{ 

for($i = 0; $i < $number; $i++){ 

$tmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[$i].”"; 
S$tmprst = $admindb->ExecSQL($tmpsql,$conn); 

S$tmpna[$i] = $tmprst[O]['name'; 

S$tmpvp[$i] = $tmprst[OJ[v_price']; 

S$tmpfd[$i] = Stmprst[ojrfold]; 

S$tmptt += $tmprst[OJ['v_price] * $tmpnm[$i]; 

@$tmpsell = Stmprst[0][sell] + 1; 

$addsql = "update tb_commo set sell = ".$tmpsell." where id = ".$tmpid[$i].""; 
$addrst = $admindb->ExecSQL($addsql,$conn); 


} 

$sql.="".$formid.",".$_POST['st].",".implode(',', $tmpna).",".$_POST[snd].”",".implode(,",$tmpvp).", 
"implode(',,,$tmpfd).",".$tmptt.",".$_POST[uid].™"; 

}else if($number == 1){ 

$tmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[0].”"; 

$tmprst = $admindb->ExecSQL($tmpsql,$conn); 

$tmptt= $tmprst[0][v_price] * $tmpnm[0]; 

@$tmpsell = $tmprst[O]['sell] + 1; 

$addsql = "update tb_commo set sell = ".$tmpsell." where id = ".$tmpid[0].”"; 

$addrst = $admindb->ExecSQL($addsql,$conn); 
$sql.=".$formid.",".$_POST['fst].",".$tmprst[O][name'].”",".$_POST['snd'].",".$tmprst[O][v_price].”, 
"Stmprst[OJ[fold].",".$tmptt.",".$_POST[uid]."™"; 

}else{ 

echo 'error’; 

exit(); 

} 
$sql.="".$_POST[taker].",".$_POST[address]",".$_POSTItel].",".$_POST[code].",".$_POST[pay]"， 
"$_POST[del]."".date("Y-m-d H:i:s").",0)"; 

S$InsertSQL = $admindb->ExecSQL($sql,$conn); 

if(false == $InsertSQL){ 

echo "<script>alert(' 购 买 失败 ');history.back;</script>"; 

}else{ 


PHP 典型 模块 开发 全 程 实录 


$updsql = "update tb_user set consume=". $tmptt.",shopping=" where name = ".$_POST[Tuid].""; 

$updrst = $admindb->ExecSQL($updsql,$conn); 

echo "<script>top.opener.location.reload();</script>"; 

echo "<script>open(forminfo.php?fid=$formid'，”，blank', width=750 height=650',false);window.close();</script>"; 


} 
人 


8.4.5 生成 订单 


订单 处 理 成 功 之 后 , 将 跳 转 到 forminfo.php 页 输出 生成 的 订单 。 订 单 生 成 的 关键 是 forminfo.php 和 
forminfo.tpl。 

在 forminfo.php 文件 中 ， 根 据 超 链 接 传递 的 订单 号 ， 查 询 订 单 表 中 的 数据 ， 并 且 将 查询 结果 赋 给 
指定 的 模板 变量 ， 最 终 指定 模板 页 forminfo.tpl。 关 键 代码 如 下 : 


<?php 

header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
include_once("system/system.inc.php"); // 包 含 类 的 实例 化 文件 
$ddnumber=$_GET['fid"]; // 获 取 订 单 号 

$sql = "select * from tb_form where formid='".$_GET[fid].""; /定义 SQL 语句 
$formarr = $admindb->ExecSQL($sql,$conn); /| 执行 SQL 语句 
$commname = explode(',,$formarr[ol[rcommo_name]); // 读 取 数 据 库 中 数据 
$commnumber = explode(,,$formarr[ol[rcommo_num]); 

$commagoprice = explode(',,$formarr[0][agoprice']); // 读 取 数 据 库 中 数据 
$commfold = explode(',,$formarr[0l[fold]); 

$smarty->assign(formarr,$formarr[0]); // 将 数据 赋 给 模板 变量 


$smarty->assign(commname',$commname); 
$smarty->assign(commnumber,$commnumben); 
$smarty->assign('commagoprice',$commagoprice); 
$smarty->assign('commfold', $commfold); 
if($formarr[OJ["del_method"]==" 平 邮 "){ 
$yprice="200"; 

}else{ 

$yprice="300"; 


} 

…// 省 略 部 分 代码 

$smarty->assign('title', 订单 提交 成 功 '); 

$smarty->display(forminfo.tpl); // 指 定 模板 页 
ww 


在 forminfo.tpl 模板 页 中 ， 获 取 模 板 变 量 传递 的 值 ， 完 成 订单 信息 的 输出 ， 并 且 创 建 订 单打 印 、 打 
印 预览 和 支付 宝 支付 的 按钮 。 关 键 代 码 如 下 : 


<table width="650" border="0" align="center cellpadding="0" cellspacing="0"> 

<tr> 

<td height="30" colspan="5" align="center" valign="middle" class="first"> 订 单 内 容 </td> 
</tr> 

<tr> 

<td width="100" height="25" align="center" valign="middle" class="left"> 商 品名 </td> 


全 
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<td width="100" height="25" align="center " valigl 
<td width="100" height="25" align="center" valign="middle" class="center"> 价 格 </td> 
<td width="100" height="25" align="center" vali middle" class="center"> 价 格 折 率 </td> 
<td width="100" height="25" align="center" valign="middle" class="right"> 合 计 </td> 
</tr> 
{foreach key=key item=item from=$commname} 
<tr> 
<td height="25" align="center" valign="middle" class="left">{$item}</td> 
<td height="25" align="center" valign="middle" class="center">{$commnumber[$key]}</td> 
<td height="25" align="center" valign="middle" class="center">{$commagoprice[$key]}&nbsp; 元 </td> 
<td align="center" valign="middle" class="center">{$commfold[$key]}&nbsp; 折 </td> 
<td align="center" valign="middle" class="right">{$commagoprice[$key]J*$commnumber[$key]}&nbsp; 元 </td> 
</tr> 
{/foreach} 
<tr> 
<td colspan="5" height="25" align="right" valign="middle"> 总 消费 ， {$formarr.total}&nbsp; 元 </td> 
</tr> 
</table> 
<input type="button" value=" 打 印 预览 " onclick="printview()" class="btn" />&nbsp;&nbsp; 
<input type="button" value=" 我 要 打印 " onclick="prints()" class="btn" /> &nbsp;&nbsp; 
<!--<input type="button" value=" 我 要 打印 " onclick="window.print()" class="btn" />--> 
<!-- 支 付 宝 支付 的 接口 操作 ， 提 交 的 数据 -> 
<a href="{$link}"> 支 付 宝 支付 </a> 


< 一 一 一 > 


middle" class="center"> 数 量 </td> 


~ 
SS 本 模块 中 的 订单 打印 、 打 印 预 览 和 支付 宝 支付 功能 都 是 在 forminfo.php 和 forminfo.tpl 文 
件 中 完成 的 ， 有 关 这 些 内 容 将 在 8.4.6 节 中 进行 详细 介绍 。 


8.4.6 ”订单 预览 、 打 印 


在 线 支付 模块 中 应 用 WebBrowser 技术 实现 订单 的 打印 和 预览 功能 .打印 和 预览 的 运行 效果 如 图 8.12 
和 图 8.13 所 示 。 


横 Microsof Xps Doc 


四 打印 到 文件 [着 和 项 和) 


[ER 


图 8.12 订单 打印 


2 


辐 轩 置 ”PHP 典型 模块 开发 全 程 实录 


3 1357695207 2013-01-09 013327 


下 单 人 : mr 收 信人 ， 明日 
邮编 : 130031 电话 : 13604357896 
地 址 : 吉林 长 春 
送 货 方 式 : 快递 付 蔓 方式 : 银行 转账 
订单 内 容 
商品 名 数量 价格 价格 折 率 合计 
数码 相机 人 16992 元 9 折 16992 元 


总 消费 : 16992 元 


项 喜 您 ! 订单 提交 成 功 。 
请 您 在 一 周 内 按 您 的 支付 方式 进行 汇款 ,汇款 时 注 明 念 的 订单 号 汇 教 后 清 及 时 通知 我 们 。 
注意 : 请 记 住 订单 号 。 以 便 日 后 查询 及 有 疑问 8 使 用 。 


打印 祯 点 我 要 打印 


图 8.13 打印 预览 


本 模块 中 的 打印 和 预览 功能 在 forminfo.tpl 模板 页 中 实现 。 首 先 载 入 printjs 脚本 文件 ， 然 后 定义 
object 标签 ， 最 后 创建 打印 预览 和 打印 的 提交 按钮 ， 并 且 通 过 onclick 事件 调用 对 应 的 方法 完成 对 应 的 
操作 。 关 键 代码 如 下 : 


<script language="javascript" src="js/print.js"></script> 

<object ID="WebBrowser1' CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></object> 
<input type="button" value=" 打 印 预览 " onclick="printview()" class="btn" /> 

<input type="button" value=" 我 要 打印 " onclick="prints()" class="btn" /> &nbsp;&nbsp;<! 一 

<input type="button" value=" 我 要 打印 "onclick="window.print()" class="btn" /> 一 > 


printjs 脚本 文件 中 封装 了 printview0 和 prints() 两 个 方法 , 分别 调用 WebBrowser 控件 中 的 方法 完成 
打印 预览 和 打印 的 操作 ， 其 代码 如 下 : 


function printview(X 

document.all.WebBrowser1.ExecWB(7,1); 1/ 打印 预览 
window.close(); 

3 

function prints(X 

document.all.WebBrowser1.Execwb(6,1); /打印 
window.close(); 


} 


@ 
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8.5 订单 查询 


8.5.1 订单 查询 功能 概述 
订单 查询 功能 根据 提交 的 订单 号 , 从 数据 库 中 查询 出 指定 订单 的 数据 。 其 运行 结果 如 图 8.14 所 示 。 


和 
明日 购物 商城 4 如。 新 年 购物 TE 


http /www mrbook com 


人 中心 新 元 彩电 | 


”超人 和 值 大 甩卖 


月 购买 和 9 本 以 上 的 彩电 ， 全 部 赠送 2 可 影 电 


网 站 公告 
欢迎 新 老 朋 友 光 痢 mm 
a 查询 订单 
口 买 100 送 100 
口 新 增 硕 目 查 词 用 户 : 查询 订单 号 : |1298863843 查询 
查询 结果 
订单 号 订货 会 员 收 贷 人 订单 全 富 付 吉 方 式 。“ 收 束 方 式 订单 状态 
客服 热线 : 1298863843 mr 潘 滨 142452 支付 宝 送 货 上 门 未 作 处 理 
"| 一 站 
NG) 400-675-1066 
图 8.14 订单 查询 


8.5.2 ”订单 查询 功能 的 实现 


订单 查询 功能 由 4 个 文件 组 成 ， 分 别 是 queryform.php、queryform.tpl、queryform.js 和 query.php。 
(1) queryform.php 文件 为 动态 PHP 文件 ， 用 于 指定 模板 页 queryform.tpl。 
(2) queryform.tpl 模板 页 用 于 创建 订单 查询 的 表单 。 载 入 JavaScript 脚本 ， 通 过 JavaScript 脚本 中 
的 queryrst() 方 法 调用 query.php 完成 订单 的 无 刷新 查询 ， 将 查询 结果 在 queryform.tpl 模板 页 中 输出 。 
queryform ,tpl 模板 页 的 关键 代码 如 下 : 


<form id='queryform' name='queryform' method='post action='#'> 

<tr> 

<td height='25' colspan='5' align='center valign='middle' class='first> 查 询 订 单 </td> 
</tr> 

<tr> 

<td width='80' height='25' align=right valign='middle' class='left> 查 询 用 户 : </td> 
<td width='130' height='25' align='eft valign='middle' class='center>&nbsp; 

<input id='name' name='name' type='text class='txt /></td> 
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<td width='100' height='25' align='right' valign='middle' class='center> 查 询 订 单 号 ，</td> 

<td width='130' height='25' align='eft valign='middle' class='right>&nbsp; 

<input id='formid name='formid' type='text' class='txt /></td> 

<td width='100' align='center valign='middle' class='center> 

<input id='enter name='enter type='button' value=' 查 询 ' class='btn' onclick='return queryrst(queryform)/></td> 
</tr> 

<tr> 

<td height='25' colspan='5' align='center valign='middle'>&nbsp;</td> 

</tr> 

</form> 


(3) 创建 queryform.js 脚本 文件 ， 定 义 queryrst0 和 showfm() 方 法 ， 调 用 query.php 文件 完成 订单 
的 无 刷新 查询 ， 并 且 将 查询 结果 赋 给 exam。 代 码 如 下 : 


// JavaScript Document 


function queryrst(form) { 

var name = form.name.value; // 获 取 表 单 提交 的 值 

var formid = form.formid.value; 1/ 获取 表 单 提交 的 值 
if(name ==" && formid == ")f // 判 断 提交 的 值 是 否 为 空 


alert(' 用 户 或 订单 号 至 少 有 一 个 不 能 为 空 '); 

form.name.focus(); 

return false; 

} 

var url = "query.php?vendee="+name+"&formid="+formid;// 定 义 调用 的 URL 
xmlhttp.open("GET",urltrue); /执行 调用 操作 
xmlhttp.onreadystatechange = showfm; /回调 函数 
xmilhttp.send(null); /执行 

} 

function showfm(){ 

if(xmlhttp.readyState == 4){ // 判 断 返回 结果 
var msg = xmlhttp.responseText; 

if(msg =='0'}{ 

exam.innerHTML = "; 

alert(' 没 有 查询 结果 '); 

jelsef 

exam.innerHTML = msg; /将 查询 结果 赋 给 exam 
} 

} 

} 


(4) 创建 query.php， 根 据 超 链 接 传递 的 参数 值 执行 订单 查询 操作 ， 并 且 返 回 查 询 结 果 。 代 码 如 下 


<?php 

header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
include_once("system/system.inc.php"); // 载 入 类 的 实例 化 文件 ， 加 载 对 象 
$vendee =$_GET[vendee]; // 获 取 超 链接 传递 的 值 

S$formid = $_GET[formid]; // 获 取 订 单 号 

S$reback = '0'; 


$tmp = "<table width='600' border='0' align='center cellpadding='0' cellspacing='0> 
<tr><td height='25' colspan='7' align='center valign='middle' class='first> 查 询 结果 </td></tr> 
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<tr><td width='100' height='25' align='center valign='middle' class='left> 订 单 号 </td> 

<td width='75' height='25' align='center valign='middle' class='center> 订 货 会 员 </td> 

<td width='75' height='25' align='center valign='middle' class='center> 收 货 人 </td> 

<td width='100' height='25' align='center valign='middle' class='center> 订 单 金额 </td> 

<td width='75' height='25' align='center valign='middle' class='center> 付 款 方 式 </td> 

<td width='75' height='25' align='center valign='middle' class='center'> 收 款 方 式 </td> 

<td width='100' height='25' align='center valign='middle' class='right> 订 单 状态 </td></tr>"; 
$sql = "select id,formid,vendee,taker,total,pay_method,del_method ,state from tb_form 
where vendee = ".$vendee." or formid = ".$formid.”"; 


$rst = $admindb->ExecSQL($sql,$conn); /| 执行 查询 语句 
if($rst)f 
foreach($rst as $value}{ // 循 环 输出 查询 结果 


$tmp .= "<tr><td height=25 align=center valign=middle class=left>".$value[formid]."</td> 
<td align=center valign=middle class=center>".$value['vendee']."</td> 

<td align=center valign=middle class=center>".$value['taker]."</td> 

<td align=centervalign=middle class=center>".$value['total]."</td> 

<td align=centervalign=middle class=center>".$value[pay_method]."</td> 
<td align=center valign=middle class=center>".$value['del_method']."</td> 
<td align=center valign=middle class=right>"; 

Switch ($value['state']X{ 

case 0: 

$tmp .= ' 未 作 处 理 '; 

break; 

case 1: 

$tmp .= ' 已 付款 '; 

break; 

case 2: 

$tmp .= ' 已 发 货 ; 

break; 

case 3: 

$tmp .= ' 已 收 货 '; 

break; 

} 

$tmp .= "</td></tr>"; 
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$tmp .= "</table>"; 

$reback = $tmp; 


} 
echo $reback; 
了 


8.6 技术 提炼 


8.6.1 收 货 人 信息 验证 


为 了 保证 网 上 交易 的 顺利 完成 ， 在 处 理 收 货 人 提交 的 信息 时 ， 必 须 对 其 填写 的 信息 进行 有 效 的 验 
证 ， 保 证 其 填写 的 信息 是 合理 有 效 的 ， 避 免 格式 上 的 错误 出 现 。 这 里 在 验证 收 货 人 提交 的 信息 时 应 用 


_ 国 
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的 是 JavaScript 脚本 和 正则 表达 式 。 旦 全 改作 信和 息 的 运行 结果 如 图 8.15 所 示 。 


[Bas -windows Intemet Ecplorer 


全 http://192.168.1.59/SOFT/05/sette.php?7uid=mrarfst=20,21snd=1,1 


邮编 :130025 


TI 重 、 蕊 100% ~ 
= = = 


图 8.15 ”验证 收 货 人 信息 的 运行 结果 


(1) 验证 手机 号 码 格式 
验证 电话 号 码 主要 是 通过 JavaScript 脚本 的 自 定义 函数 调用 正则 表达 式 ， 实 现 对 手机 号 码 的 验证 。 
编写 的 判断 手机 号 码 格式 的 正则 表达 式 如 下 : 


/^13(\d{9))$I^15(\d{9})$,; 


其 判断 手机 号 码 是 否 是 以 13 或 者 15 开始 ， 并 判断 是 否 是 11 位 数字 。 判 断 手 机 号 码 格式 是 否 正 确 
的 自 定义 函数 的 代码 如 下 : 


function checkregtel(regtel){ 

var str=regtel; 

var Expression=/^13(\d{9})$|^15(\d{9})$/;// 判 断 手机 号 码 的 号 段 和 格式 是 否 正确 
Var objExp=new RegExp(Expression); 

if(objExp.test(str)==true){ 

return true; 


}else{ 


return false; 

} 

ER 

(2) 验证 收 货 人 的 其 他 信息 

验证 收 货 人 的 其 他 信息 主要 应 用 的 是 JavaScript 脚本 ， 通 过 JavaScript 脚本 中 的 自 定义 函数 获取 表 
单 提交 的 数据 ， 在 自 定义 函数 中 判断 数据 是 否 合理 ， 如 果 不 合理 ， 则 通过 div 标签 直接 输出 错误 提示 。 
验证 收 货 人 的 其 他 信息 的 代码 如 下 : 

function chkreginfo(form,mark,editX{ 

if(mark==0 || mark=="all"}{ // 验 证 收 货 人 

if(form.taker.value=="" 


chknew_taker.innerHTML=" 请 输入 收 货 人 !"; 
form.taker.style.backgroundColor="#FF0000"; 
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return false; 
}else{ 
chknew_taker.innerHTML=""; 
form.taker.style.backgroundColor="#FFFFFF"; 
} 

} 

if(mark==1 || mark=="all"}{ // 验 证 邮编 
if(form.code.value=="){ 
chknew_code.innerHTML=" 请 输入 邮编 !"; 
form.code.style.backgroundColor="#FF0000"; 
return false; 
}else if(isNaN(form.code.value)){ 
chknew_code.innerHTML=" 邮 编 由 数字 组 成 !"; 
form.code.style.backgroundColor="#FF0000"; 
return false; 
}else if(form.code.value.length!=6\{ 
chknew_code.innerHTML=" 邮 编 由 6 位 数字 组 成 ! "; 
form.code.style.backgroundColor="#FF0000"; 
return false; 
}else{ 
chknew_code.innerHTML=""; 
form.code.style.backgroundColor="#FFFFFF"; 
} 


} 

if(mark==2 || mark=="all")f // 验 证 电话 号 码 
if(form.tel.value==""){ 
chknew_tel.innerHTML=" 请 输入 电话 号 码 !"; 
form.tel.style.backgroundColor="#FF0000"; 

return false; 

}else if(Icheckregtel(form.tel.value)\{ 
chknew_tel.innerHTML=" 电 话 号 码 的 格式 不 正确 ! "; 
form.tel.style.backgroundColor="#FF0000"; 

return false; 

}else if(isNaN(form.tel.value)){ 
chknew_tel.innerHTML=" 电 话 号 由 数字 组 成 !"; 
form.tel.style.backgroundColor="#FF0000"; 

42 return false; 

}else{ 

chknew_tel.innerHTML="™"; 
form.tel.style.backgroundColor="#FFFFFF"; 

} 


re | mark=="all")f /验证 联系 地 址 
if(form.address.value=="){ 
chknew_address.innerHTML=" 请 输入 联系 地 址 ! "; 
form.address.style.backgroundColor="#FF0000"; 

return false; 


}else{ 


chknew_address.innerHTML="™"; 
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form.address.style.backgroundColor="#FFFFFF"; 
} 
} 
. 
8.6.2 ”动态 生成 订单 号 
动态 生成 订单 号 应 用 的 是 time() 函 数 ,该 函数 获取 当前 的 UNIX 时 间 戳 ,返回 值 为 从 UNIX 纪元 ( 格 
林 威 治 时 间 1970 年 1 月 1 日 00:00:00) 到 当前 时 间 的 秒 数 。 该 函数 的 语法 如 下 ;: 


inttime ( void ) 


time() 函 数 没 有 参数 ， 返 回 值 为 UNIX 时 间 戳 的 整数 值 。 
这 里 就 是 应 用 time() 函 数 返回 的 UNIX 时 间 戳 作为 订单 号 的 。 在 订单 处 理 页 settle_chk.php 中 ， 通 
过 time() 函 数 获取 UNIX 时 间 戳 的 整数 值 ， 并 且 将 其 赋 给 $formid 变量 作为 订单 号 。 关 键 代码 如 下 : 


$formid=time(); // 生 成 订单 号 
8.6.3 WebBrowser 打印 


在 线 支付 模块 中 订单 的 打印 和 预览 是 非常 重要 的 两 个 功能 ,实现 这 两 个 功能 主要 应 用 的 是 WebBrowser 
打印 技术 。 打 印 预览 的 运行 结果 如 图 8.16 所 示 。 


订单 号 ;1357695207 订单 时 间 : 2013.01.09 013327 
下 单 人 : mr 收 货 人 : 明日 中 
邮编 : 130031 电话 : 13604357896 
地 址 ; 吉林 长 春 
送 做 方式 : 快递 付 壹 方式 : 松 行 转账 
订单 内 容 
数量 价格 价格 折 率 合计 
1 16992 元 9 折 16992 元 
总 消费 : 16992 元 


蕉 喜 您 ! 订单 提交 成 功 。 
请 您 在 一 周 内 按 您 的 支付 方式 进行 汇款 ,汇款 时 注 明 您 的 订单 号 | 汇款 后 请 及 时 通知 我 们 。 
注意 : 请 记 住 订单 号 。 以 便 日 后 查询 及 有 经 癌 时 使 用 * 


打印 祯 点 我 要 打印 支付宝 支付 


图 8.16 ”打印 预览 的 运行 结果 
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WebBrowser 是 正 内 置 的 浏览 器 控件 ， 无 须 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 目标 文档 的 
生成 ， 减 轻 服 务 器 负荷 ; 缺点 是 源 文档 的 分 析 操 作 复杂 ， 并 且 对 源 文档 中 要 打印 的 内 容 进 行 约束 。 
订单 打印 和 预览 主要 应 用 正 内 置 的 WebBrowser 控件 ， 该 控件 的 具体 参数 如 表 8.1 所 示 。 


表 8.1 IE 内置 的 WebBrowser 控件 的 参数 说 明 


参数 说 了 明 
document.all.WebBrowser.Execwb(7,1) 表示 打印 预览 
document.all.WebBrowser.Execwb(6,1) 表示 打印 
document.all.WebBrowser.Execwb(6,6) 表示 直接 打印 
document.all.WebBrowser.Execwb(8.1 表示 页 面 设置 
document.all.WebBrowser.Execwb(1,1) 打开 页 面 
document.all.WebBrowser.Execwb(2,1) 关闭 所 有 打开 的 IE 窗口 
document.all.WebBrowser.Execwb(4,1) 保存 网 页 
document.all.WebBrowser.Execwb(10.1 查看 页 面 属性 
document.all.WebBrowser.Execwb(17,1) 全 选 
document.all.WebBrowser.Execwb(22.1 刷新 
document.all.WebBrowser.Execwb(45.1 关闭 窗 体 无 提示 


8.6.4 支付 宝 支付 


本 模块 中 通过 支付 宝 完成 在 线 支付 。 要 使 用 支付 宝 进行 在 线 支 付 ， 必 须 先 在 支付 宝 中 注册 一 个 支 
付 宝 账 户 ， 然 后 下 载 支付 宝 提 供 的 接口 文件 。 通 过 支付 宝 提供 的 接口 文件 实现 电子 商务 网 站 与 支付 宝 
的 交互 。 

在 支付 宝 提供 的 接口 文件 中 ， 提 供 了 完整 的 实例 程序 ， 只 要 依照 实例 程序 中 的 操作 方法 ， 将 电子 
商务 网 站 中 指定 的 数据 提交 到 支付 宝 提供 的 数组 参数 中 ， 然 后 支付 宝 会 通过 自 定义 的 方法 将 数据 提交 
到 指定 的 网 站 中 。 

(1) 在 forminfo.php 文件 中 ， 根 据 从 数据 库 中 读 取 的 数据 ， 定 义 支付 宝 的 数组 参数 ， 其 关键 代码 
如 下 : 


require_once("alipay_service.php"); 

require_once("alipay_config.php"); 

$parameter = array( 

"service" => "trade_create_by_buyer"， /交易 类 型 , 必 填 实物 交易 等 于 trade_create_by_buyer( 需 要 填写 物流 ) 


"partner" =>$partner, // 合 作 商 户 号 

"return_url" =>$return_url, // 同 步 返 回 

"notify_url" =>$notify_url, /异步 返回 

"_input_charset" => $_input_charset, // 字 符 集 ， 默 认为 GBK 

"subject" =>$formarr[0]["commo_name"], // 商 品名 称 ， 必 填 

"body" => $formarr[O]["commo_num"], // 商 品 描述 ， 必 填 

"Out_trade_no" => $ddnumber, // 商 品 外 部 交易 号 ， 订 单 号 ， 必 填 ， 每 次 测试 都 需要 修改 
"logistics_fee"=>$yprice, // 物 流 配 送 费 用 


"logistics_payment"=>'BUYER_PAY'， ”// 物 流 配送 费用 付款 方式 : BUYER_PAY ( 买 家 支付 ) 
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"logistics_type"=>EXPRESS，， /物流 配送 方式 : POST (平邮 ) 、EMS (EMS) 、EXPRESS (其 他 快递 ) 


"price" => $formarr[OJ["total"], /商品 单价 ， 必 填 
"payment_type"=>"1", /| 默认 为 1， 不 需要 修改 
"quantity" => "1", /商品 数量 ， 必 填 
"show_url" => $show_url, /商品 相关 网 站 

"seller_ email" => $seller_email /卖家 邮箱 ， 必 填 


); 
$alipay = new alipay_service($parameter,$security_code,$sign_type); 
Slink=$alipay->create_url(); 

$smarty->assign("link", $link); 


(2) 在 forminfo.tpl 模板 页 中 创建 超 链接 ， 完 成 支付 操作 ， 其 关键 代码 如 下 : 


<!-- 支 付 宝 支付 的 接口 操作 ， 提 交 的 数据 -> 
<a href="{$link}"> 支 付 宝 支付 </a> 


<!—- > 


4 
Dr 
助 功能 的 实现 方法 。 至 于 涉及 支付 宝 提 供 的 参数 和 设置 都 是 个 人 雇 拟 的 ， 真实 的 参数 将 在 实际 的 运 
作 中 由 支付 宝 提供 。 在 实际 的 运作 中 ， 需 要 向 支付 宝 申请 一 个 账户 ， 然 后 使 用 由 支付 宝 提供 相应 的 
参数 就 可 以 应 用 此 功能 了 。 


8.7 本 章 小 结 


本 项 目 使 用 Smarty、PDO 数据 库 抽象 层 和 Ajax 等 目前 的 主流 技术 ， 操 作 一 个 在 线 支 付 模块 。 通 
过 本 章 的 学 习 ， 读 者 不 仅 可 以 了 解 在 线 支 付 模块 的 操作 流程 ， 而 且 可 以 熟悉 购物 车 、 订 单 及 加 密 技 术 
的 开发 思想 。 


全 立 
性 - 


留言 本 模块 


(PHP+ MySQL 5.0 实现 ) 


本 章 以 构建 一 个 适合 于 中 小 型 企业 应 用 的 留言 本 为 目的 ， 全 面 介 
绍 留言 本 的 设计 思路 及 流程 。 在 介绍 留言 本 开发 过 程 的 同时 ， 对 PHP 
中 一 些 常 用 函数 及 编程 技巧 也 进行 了 详细 介绍 ， 读 者 不 仅 可 以 对 留言 
本 的 开发 过 程 及 思路 有 个 全 面 掌 担 , 而 且 还 可 以 学 习 到 许多 PHP 的 编 
程 技巧 ， 从 而 金 面 提高 个 人 的 基础 及 编程 能 力 。 留 言 本 的 开发 过 程 虽 
然 较为 简单 ， 却 涵盖 了 动态 网 站 开发 的 大 部 分 经 典 功能 模块 ， 所 以 只 
要 熟练 事 担 本 章 的 内 容 ， 并 且 能 够 灵活 应 用 ， 就 可 以 开发 出 大 型 的 网 
上 论坛 、 博 客 以 及 其 他 一 些 网 络 中 较 流 行 的 网 站 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

| 掌握 留言 本 的 设计 思路 及 流程 
掌握 MySQL 数据 库 的 设计 方式 和 方法 
掌 查 PHP 连接 MySQL 数据 库 的 方法 
能 够 灵活 运用 常用 的 SQL 查询 语句 
掌握 PHP 实现 数据 分 页 显示 的 方法 


各 于 于 至 
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9.1 留言 本 模块 概述 


无 论 是 应 用 程序 还 是 Web 项 目 ， 开 发 人 员 只 有 提前 做 好 规划 并 能 够 清楚 地 掌握 项 目的 设计 思路 及 
流程 ， 才 能 开发 出 高 效 、 完 善 的 项 目 。 本 节 将 通过 留言 本 的 功能 阐述 、 功 能 结构 等 方面 介绍 留言 本 的 
设计 思路 及 实现 过 程 。 


9.1.1 模块 概述 


一 个 适合 于 个 人 或 小 规模 企业 使 用 的 留言 本 模块 ， 只 需 具 有 添加 留言 和 删除 留言 功能 即 可 。 由 于 
本 章 开 发 留言 本 是 以 适用 于 中 小 型 企业 为 出 发 点 ， 以 使 读者 全 面 掌 握 留 言 本 的 设计 流程 及 提高 个 人 的 
编程 能 力 为 目的 ， 所 以 本 章 所 介绍 的 留言 本 除 具 有 添加 留言 和 删除 留言 的 功能 外 ， 还 具有 用 户 注册 、 
用 户 登 录 、 编 辑 留言 等 功能 。 本 章 所 介绍 的 留言 本 运行 效果 如 图 9.1 所 示 。 


今天 是 : 2011 年 5 月 日 性 基 二 多 条 的 位 秆 :明日 久 起 》》 并 页 
用 加 主题 :五 -性 
有 PS: [一 
= 
这 SE 
万 历 全 条 由 Pt 址 肥 0ict 
日 -二 三 四 五 六 


gown un 
1 61 1019 20 2 


2 A 的 委 洒 负 轩 交 出 版 ， 请 关注 4 
29 30 at | 
! 
到 zn s 到 局 信箱 黑 I 地 址 风 0ict ; 


六 新 留言 三 是; 鲍 汪 词 费 件 么 时 候 上 市 和 ?了 
， 匾 程 辣 稀 什么 时 . [11/05/03] 
， 篇 程 间 风 即 梅 出 . [tl705/03] 
， 五 一 快乐 [11/05/03] 


A :请 冯 编程 问 拱 什 么 时 候 上 市 季 ? 
Er | 
总 信 箱 黑 卫 二 让 cic ; 
共有 留言 3 条 每 页 显示 3 条 第 1 页 / 共 1 页 首页 上 -页 下 -页 尾 页 
T 胃 日 和 言 本 所 权 所 有 寺 认 省 明日 科技 有 限 公司 1 未 从 括 机 禁止 揽 制 或 于 立 村 全 1 
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图 9.1 留言 本 运行 效果 
9.1.2 功能 结构 


在 开发 程序 前 ， 先 规划 一 下 留言 本 模块 的 功能 结构 ， 这 样 在 开发 程序 时 可 以 保证 思路 清晰 。 留 言 
本 模块 的 功能 结构 如 图 9.2 所 示 。 


他- 
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[人 
〈 溯 崩 践 ) 蔬 归 天 事 


图 9.2 留言 本 模块 的 功能 结构 图 
9.1.3 程序 预览 
留言 本 模块 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 模块 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 几 个 典 


型 功能 的 页 面 ， 其 他 页 面 请 参见 光盘 中 的 源 程序 。 
留言 本 首页 页 面 运行 结果 如 图 9.3 所 示 。 


今天 是 5211 和 5 月 5 日 星 类 二 悠 当前 的 位 置 ; 明 忆 当 百 本 > 首 页 


用 本 有 手 题 :五 -全 和 
mr: 
"mm 
| 2 和 
a 售 煌 开工 地 址 AA0ica ; 
日 一 二 三 四 | 
二 二 贡生 全 
0 
好 A 人- 枕 本 词 生 即 检 出 版 ， 清 关注 
3 0 3 | 
下 wm 5 加 芒 箱 量 匡 地 址 银 oica ; 
a Ed 
， 区 术 记 风 慎 和 时 . [ilyesfoa] [ 
， 总 量词 由 加 出 iVIS1OD] bd ! 
”五 快乐 HSA09] 这 请问 用 订 血 什么 时 位 上 市 和 了 


运 
3 信行 本地 址 忆 oica ; 
共有 留言 3 条 每 再 呈 示 3 条 第 1 页 / 共 1 页 首页 上 -页 下 -机 尾 页 


[图 日 等 言 节 」 后台 所有 吉林 省 日 科 技 有 素 公 司 | 未 径 检 取 禁止 揽 和 下 当 立 鲁 等 1 


9.3 留言 本 首页 页 面 运行 结果 
用 户 注册 页 面 运行 结果 如 图 9.4 所 示 。 用 户 只 有 注册 成 功 后 才 可 以 获取 登录 用 户 名 和 密码 , 进行 登 


国 
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录 后 ， 才 能 执行 发 表 留言 等 操作 。 
”用 户 注册 
注册 声明 

破坏 完 法 和 法 律 、 行 政法 规 实 族 的 ; 


《 2 ) 坊 动 着 路 国家 政权 ， 推 基 社 会 主义 制度 的 
3 提 东 时 人 散布 语言 ， 扰 彰 社 会 秩序 的 ， 


性 别 : 页 了 

reai] 地 址 :FinerisoEsingiseEt co 
联系 电话 : SEE 

区 号 码 ; 455845 

头 信 选 择 Te 司 及 


图 9.4 用 户 注册 页 面 运行 结果 
发 表 留 言 页 面 运行 结果 如 图 9.5 所 示 。 查 找 留言 页 面 运行 结果 如 图 9.6 所 示 。 


调和 择 查 找 式 : [下 要” 司 历 一 Ez] 


圭 题 : 五 一 快乐 
年 这 者 朋 到 们 五 一 所 使节?T FF 
ee 目 痘 | 视 所 有 染 者 朋友 们 五 一 节 快 乐 b 
五 党 信 条 型 rrtt 直 有 oica : 
图 9.5 发 表 留言 页 面 运 行 结果 图 9.6 查找 留言 页 面 运行 结果 


9.2 数据库 和 数据 表 设 计 


9.2.1 数据 库 设 计 


留言 本 模块 采用 的 是 MySQL 数据 库 ， 主 要 用 于 存储 用 户 信息 和 留言 信息 。 这 里 将 数据 库 命 名 为 
db_ messagebook， 其 中 包含 的 数据 表 如 图 9.7 所 示 。 
轩 服务 只: localhost ， 局 数据 库 : db_messagebook 


表 扣 作 记录 教 人 @ 。 类 型 整理 襄 明 
也 leaveword 园 辐 图 芋 硬 头 1 MYSAM ”9b2312_chinese_c 。 留言 信 息 表 
th_user 国 加 兰 国 X 5 MSAM gb2312_chinese_ci 。 用 户 信息 夹 

2 个 表 训 7 MyISAM gb2312_chinese_ci 


图 9.7 数据 库 结构 


9.2.2 ”数据 表 设 计 
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数据 库 设计 完成 后 ， 下 面 来 看 各 个 数据 表 的 结构 和 字段 说 明 。 
用 户 信息 表 主 要 用 于 存储 用 户 信息 ， 数 据 表 字 段 设计 如 图 9.8 所 示 。 


固 服务 器 : localhost 局 孝 所 库 :db_messagebook 》 国 表 : tb_user 
字段 雪 型 整理 属性 hul 睹 认 藤 明 

te) 百 aulo_ncrement 入 
useme varchar50) gt2312_chinese_c 否 用 户 g 称 
tmename varchar(50) gt2312_chinese_ci 要 真实 名 称 
emal varchar50) gt2312 chnese_c 百 邮 闪 
qq varchar(20) «gt2312_chinese_ci 否 六 
ol varchar@0) gt2312_chinese_ci 否 电话 
np varchar(10) gt2312_chinese_ci 至 Tm 
address 。 varcharf250) gt2312_chinese_ci 否 地 址 
ace varchars0) 。 962312_chnese_ol 否 六 
regtime dateime 至 发 有 时 间 
aex Varchar2) gt2312_chinese_ci 和 全 别 
usertype ink(2) 百 如 限 
serpwd 。 varchar50) gt2312_chinese_ci 否 苹果 宙 凤 


图 9.8 ”用户 信息 表 


留言 信息 表 主 要 用 于 存储 用 户 留言 信息 ， 数 据 表 字 段 设计 如 图 9.9 所 示 。 


画 服务 器 :Tocalhost》 夯 数 晤 库 : db_messagebooky 国 表 :tp_ieaveword 
字 度 天 型 整理 属性 Null 歌 认 新 外 EE] 
uu int) 否 ato_inaement 留 疝 ia 
userd ne) 否 和 
createtime datatime 理 留言 H 间 
te varchar(250) gb2312_chinese_ci 至 留言 主题 
contont toxt 9b2312_chinoeso_ci 否 留言 内 雁 
图 9.9 留言 信息 表 
9.2.3 ”连接 数据 库 


由 于 模块 大 部 分 页 面 都 需要 使 用 数据 库 ， 如 果 每 页 都 编写 相同 的 数据 库 连 接 代 码 ， 会 显得 十 分 繁 
琐 。 所 以 本 模块 将 数据 库 连 接 代码 单 独 存 入 一 个 文件 conn.php 中 ， 在 需要 与 数据 库 连 接 的 页 面 中 ， 使 


用 包含 语句 包含 conn.php 文件 即 可 。 该 模块 实现 与 数据 库 连 接 的 代码 如 下 


<?php 


$conn=mysql_connect("localhost", "root","111"); // 连 接 数据 库 服务 器 
mysql_select_db("db_messagebook",$conn); // 选 择 数据 库 
mysql_query("set names gb2312"); // 设 置 页 面 编码 格式 


?> 


Ln 


9.3 发 表 留 


9.3.1 发 表 留 言 概述 
用 户 登录 成 功 后 ， 单 击 导航 菜单 中 的 “发 表 留言 ” 超 链 接 ， 进 入 发 表 留 言 页 面 ， 填 写 留言 主题 及 


号 轩 兽 ”PHP 典型 模块 开发 全 程 实录 


留言 内 容 ， 单 击 “ 发 表 ” 按 钮 ， 即 可 发 表 留 言 。 发 表 留 言 页 面 的 运行 结果 如 图 9.10 所 示 。 


- 骨 且 留言 本 一 一 一 


今天 是 : ”2011 年 5 月 3 日 ”星期 二 您 当前 的 位 置 : 明日 留言 本 >》 发 表 留 言 
用 户 登 录 发 表 留 言 


再 这 大 丽 友 人 五 一 人 东 ?| | 


15 16 17 18 19 20 21 Wes 
2 2324 2522 28 


29 30 31 
司 z s 到 


最 新 留言 
， 编程 词典 什么 时 , [11/05/03] 
， 编程 词典 即 格 出 . [11/05/03] 发 表 | ”年 写 
， 五 一 快乐 [11/05/03] 


『 明 日 留言 本 】 版 权 所 有 吉林 省 明日 科技 有 限 公司 | 未 经 授权 禁止 复制 或 建立 镇 便 | 
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图 9.10 发 表 和 留言 页 面 的 运行 结果 
9.3.2 发 表 留 言 页 面 设计 


一 个 优秀 的 Web 程序 ， 不 仅 应 具有 合理 的 代码 编写 规则 和 较 高 的 代码 执行 效率 ， 合 理 的 页 面 设计 
方式 和 美观 的 页 面 也 是 不 可 缺少 的 。 为 了 保证 整个 留言 本 页 面 的 一 致 性 ， 在 设计 页 面 时 ， 将 留言 本 的 
头 部 内 容 存储 在 top.php 文件 中 ， 将 用 于 显示 版 权 信息 的 尾部 内 容 存储 在 bottom.php 文件 中 。 这 样 ， 在 
新 建 留言 本 的 功能 页 面 时 ， 只 需 在 页 面 最 上 方 加 上 include_once("top.php") 和 在 页 面 最 下 方 加 上 
include_once("bottom.php") 即 可 。 应 用 这 种 页 面 设计 方式 ， 还 可 以 提高 程序 的 开发 效率 和 易 维 护 性 。 


攻 5 为 了 提高 网 络 的 传输 速度 ， 尽 量 将 页 面 图 片 存储 为 .gif 或 jpg 格式 。 另 外 ， 具 有 美观 得 体 
的 页 面 也 是 开发 人 员 所 必须 考虑 的 因素 之 一 。 


LV 
说 明 之 所 以 在 设计 页 面 时 ， 将 图 片 存储 为 .gif 或 .jpg 格式， 是 因为 这 两 种 格式 图 片 具 有 所 占 空 
间 小 、 画 面 质量 高 等 特点 ， 从 而 可 以 有 效 提高 网 站 的 传输 率 。 


@ 
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发 表 留 言 页 面 的 设计 效果 如 图 9.11 所 示 。 


图 9.11 发 表 留 言 页 面 的 设计 效果 
发 表 留 言 页 面 的 设计 流程 如 下 : 
(1) 应 用 include 语句 引用 顶部 Banner 广告 头 文件 top.php 页 ， 代 码 如 下 : 


<?php include("top.php"); ?> 


(2) 应 用 include 语句 引用 左 侧 功 能 导航 文件 left.php 页 ， 代 码 如 下 : 
<?php include("left.php"); ?> 


(3) 为 了 使 页 面 效 果 更 丰富 多 彩 、 更 人 性 化 ， 分 别 定义 文本 输入 框 和 按钮 的 CSS 样式 来 制作 个 
性 化 的 表单 。 文 本 输入 框 和 按钮 的 CSS 样式 代码 如 下 : 


TS > 


< -一 一 一 一 一 一 一 一 - 
buttoncss { 
font-family: "Tahoma", "宋体 "; 

font-size: 9pt; color #FCC42C; 

border: 1px #003399 solid; 

color:006699; 

BORDER-BOTTOM: #F CC42C 1px solid; 
BORDER-LEFT: #FCC42C 1px solid; 
BORDER-RIGHT: 证 CC42C 1px solid; 
BORDER-TOP: #F CC42C 1px solid; 
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background-color: #ffffff; 
CURSOR: hand; 
font-style: normal ; 


< 定义 文本 输入 框 的 CSS 样式 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 -: > 
-inputcss { 

font-size: 9pt; 

font-family: "宋体 "; 

font-style: normal; 


. 
发 表 留 言 页 面 涉及 的 HTML 表单 的 重要 元 素 如 表 9.1 所 示 。 
表 9.1 发 表 留 言 页 面 涉及 的 HTML 表单 的 重要 元 素 


name="forml" method="post" action="saveleaveword.php" onsubmit="return 
chkinput(this. 


9.3.3 ”将 用 户 留言 内 容 保存 到 数据 库 中 


实现 发 表 留 言 功 能 ， 首 先 应 建立 用 于 填写 留言 信息 的 表单 ， 然 后 通过 PHP 脚本 获取 表单 提交 的 数据 ， 
将 留言 信息 添加 到 指定 的 数据 表 中 。 在 leaveword.php 中 完成 填写 留言 信息 表单 的 创建 ,在 saveleaveword.php 
文件 中 获取 表单 提交 的 留言 信息 ， 并 且 将 其 添加 到 指定 的 数据 表 中 ， 其 关键 代码 如 下 : 


<?php 

session_start(); 1/ 初始 化 Session 变量 

include_once("conn.php"); // 包 含 数据 库 连 接 文件 

$sql=mysql_query("select id from tb_user where usernc=".$_SESSION["unc'"]."",$conn); // 执 行 查询 语句 
$info=mysql_fetch_array($sql); // 获 取 查 询 结果 

$userid=$infof'id]; // 获 取 用 户 1D 

$createtime=date("Y-m-d H:i:s"); // 获 取 系 统 当前 时 间 

// 执 行 添加 语句 


if(mysqlL_query('insert into tb_leaveword(userid,createtime ,title,content) 
values('$userid','$createtime',".$_POSTT'title].",”".$_POST[content].")",$conn)X{ 
echo "<script>alert(' 留 言 发 表 成 功 ! ');history.back();</script>"; 

}else{ 

echo "<script>alert( 留言 发 表 失败 ! ');history.back();</script>"; 

} 


?> 


@ 
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9.4 查看 留 


LD 


9.4.1 查看 留言 概述 


单 击 导航 菜单 中 的 “查看 留言 ” 超 链接 ， 即 可 进入 查看 留言 页 面 ， 其 运行 结果 如 图 9.12 所 示 。 该 
页 面 展示 出 了 留言 主题 、 内 容 、 发 布 者 等 信息 ， 并 且 对 当前 登录 用 户 的 权限 进行 判断 ， 如 果 当 前 用 户 
是 留言 的 发 布 者 ， 那 么 还 可 以 对 留言 进行 编辑 ， 否 则 只 能 查看 留言 内 容 


用 户头 主题 :五 -快乐 训 入 
APs: : 
害 码 : 必 | 
本 | 本 XA FRE 
! 
4 万 俐 区 黑 了 地 址 由 0ica ; 
BH-==0 | 后 关 本 二 信人 
人 
a9 102 
223H 3B A 六 本 词类 印 本 出 需 ， 请 关注 + 
2 a | 
到 zm s 下 渤 革 攻 时 THh 直 怕 oiea ; 
最新 入 言 主 十: 可 各 河和 什么 时 要 上 而 季 ? 国术 
， 加 各 局 由 什么 时 .lyas/aa] 
， 鲍 程 词典 邓 桩 出 . [11J05/03] 群 上 
， 五 一 快 条 [11J05/05] 这 : 许 问 扩 和 司 秽 什么 时 供 上 市 和? 
到 | 


信箱 器 划 址 全 0c | 
失信 言 3 条 等 页 显示 3 条 第 ! 页 / 兴 1 页 首页 上 一 页 下 一 页 尾 页 
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图 9.12 查看 留言 页 面 运行 结果 
9.4.2 ”查看 留言 页 面 设计 


为 了 使 留言 者 和 留言 内 容 相互 对 应 ， 在 设计 留言 内 容 显 示 页 面 时 ， 将 留言 者 的 常规 信息 动态 显示 
在 页 面 左 侧 ， 将 留言 内 容 显 示 在 留言 页 面 的 右 侧 ， 并 通过 自 定义 函数 对 用 户 留 言 内 容 中 的 HTML 标记 
进行 过 滤 。 查 看 留言 页 面 的 设计 效果 如 图 9.13 所 示 。 


PHP 典型 模块 开发 全 程 实录 


查看 留言 页 面 的 设计 流程 如 下 : 
(1) 在 HTML 标记 中 嵌入 PHP 代码 动态 地 显示 用 户 常 规 信 息 ， 代 码 如 下 : 


<img src="images/email.gif" width="45" height="16" alt="<?php echo $info1[email];?>"/> 
<img src="images/ip.gif'" width="55" height="16" alt="<?php echo S$info1[ip];?>"/> 
<img src="images/qq.gif" width="45" height="16” alt="<?php echo Sinfo1[qq];?>"/> 


(2) 通过 自 定义 函数 unhtml0 实 现 用 户 留言 中 的 HTML 标记 原样 输出 ， 代 码 如 下 : 


function unhtml($content)f 

$content=htmlspecialchars($content); // 将 单 引号 、 小 于 号 、 大 于 号 等 原样 输出 
$content=str_replace(chr(13),"<br>", $content); /原样 输出 换行 标记 
$content=str_replace(chr(32),"&nbsp;", $content); // 原 样 输出 空格 

return trim($content); 

} 


9.4.3 ”在 页 面 中 输出 留言 信息 


在 index.php 文件 中 ， 通 过 while 语句 循环 输出 留言 信息 的 内 容 ， 并 且 应 用 分 页 技术 控制 每 页 显示 
3 条 记录 。 关 键 代 码 如 下 : 


<?php 
$sql=mysql_query("select count(*) as total from tb_leaveword ",$conn); 
$info=mysql_fetch_array($sql); 


$total=$info[total]; /获取 总 留言 条 数 

if($total==0){ // 如 果 总 留言 条 数 为 0， 则 给 出 提示 
echo "<div align=center> 对 不 起 ， 暂 无 留言 ! </div>"; 

}else{ 


// 判 断 查询 字符 串 page 的 值 是 否 为 空 ， 如 果 为 空 ， 则 默认 显示 第 一 页 
ifllisset($_GET["page"]) || !is_numeric($_GET["page"]))}{ 

$page=1; 

}else{ 

$page=intval($_GET["page"]); 

} 


$pagesize=3; // 规 定 每 页 显示 3 条 留言 
if($total% $pagesize==0X{ // 获 取 总 页 数 
$pagecount=intval($total/$pagesize); 
}else{ 

$pagecount=ceil($total/$pagesize); 


$sql=mysql_query("select * from tb_leaveword order by createtime 
desc limit ".($page-1)*$pagesize.",$pagesize ",$conn); 
while($info=mysql_fetch_array($sql)){ // 通 过 while 循环 显示 所 有 留言 
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…// 显 示 用 户 信息 和 用 户 留 言 信息 

<?php 

$sqlu=mysql_query("select usernc from tb_user where id=". $info["userid"]."",$conn); 
$infou=mysql_fetch_array($sqlu); 
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if($infou["usernc"]==$_SESSION["unc"]X // 判 断 登 录用 户 是 否 为 留言 的 发 表 者 ， 如 果 是 则 显示 “编辑 ”按钮 
J 

…/ 显 示 “ 编 辑 ” 按 钮 

<?php 


if($_SESSION["unc"]!="){ 

$sqld=mysql_query("select usertype from tb_user where usemc=".$_SESSION["unc"]."",$conn); 
$infod=mysql_fetch_array($sqld); 

if($infod["usertype"]==1){ 1/ 判断 当前 登录 用 户 是 否 为 管理 员 ， 如 果 是 则 显示 “删除 ”按钮 

?> 

<a href="deleteleaveword.php?id=<?php echo $info[id];?>" class="a1"> 删 除 </a> 

<?php 

} 

} 


?> 
…"/ 显 示 用 户 信息 及 用 户 留 言 信息 
<?php 


?> 


培 明 上 述 代码 中 ， 实 现 留言 信息 的 分 页 显示 是 通过 MySQL 数据 库 的 扩展 关键 字 limit 来 实现 
的 ， 该 关键 字 后 跟 两 个 参数 。 其 中 ， 第 一 个 参数 用 于 指定 要 显示 记录 的 起 始 位 置 ; 而 第 二 个 参数 用 
于 指定 所 要 显示 的 记录 个 数 。 


9.4.4 将 留言 信息 进行 分 页 显示 


用 户 发 表 完 留言 后 ， 可 以 通过 查看 留言 页 面 查 看 所 有 留言 内 容 。 由 于 用 户 的 留言 数目 较 多 ， 如 果 在 
同一 页 面 中 显示 所 有 留言 信息 ， 则 会 给 用 户 浏览 带 来 很 大 的 不 便 ， 所 以 通过 分 页 的 方式 显示 用 户 留 言 内 
容 是 不 错 的 选择 。 在 实现 用 户 留言 内 容 分 页 显示 时 ， 主 要 应 用 is_numeric0 函 数 判 断 用 户 通 过 GET 方法 
提交 的 数据 是 否 为 数值 型 ， 并 通过 ceil0 函 数 对 页 码 数据 进行 向 上 取 整 。 下 面 将 对 这 两 个 函数 进行 介绍 。 


1. is_numeric() 函 数 

如 果 该 函数 的 参数 为 数字 或 数字 字符 串 ， 则 返回 tue， 和 否则 返回 false。 其 语法 如 下 : 
bool is_numeric ( mixed var ) 

参数 var 为 要 进行 判断 的 数据 。 

2. ceil() 函 数 

ceil0 函 数 用 于 对 浮 点 数 进行 向 上 取 整 。 其 语法 如 下 : 


float ceil ( float value ) 
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参数 value 为 要 进行 向 上 取 整 的 数据 。 
在 显示 用 户 留言 时 ， 根 据 传 入 当前 页 面 查询 字符 串 page 的 值 来 决定 所 要 显示 的 记录 范围 。 创 建 分 
页 超 链接 的 关键 代码 如 下 : 


<table width="550" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 

<td width="351"><div align="left"> 共 有 留言 &nbsp;<?php echo S$total;?>&nbsp; 条 &nbsp; 
每 页 显示 <?php echo $pagesize;?>&nbsp; 条 &nbsp; 第 &nbsp;<?php echo $page;?>&nbsp; 
页 / 共 &nbsp;<?php echo $pagecount;?>&nbsp; 页 </div> 

</td> 

<td width="199"><div align="right"> 

<a href="<?php echo $_SERVER['PHP_SELF]?>?page=1" class="a1"> 首 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page= 

<?php 

if($page>1) 

echo $page-1; 

else 

echo 1; 

?> 

"class="a1"> 上 一 页 </a>&nbsp; 

<a href="<?php echo $_SERVER["PHP_SELF"]?>?page= 

<?php 

if($page<$pagecount) 

echo $page+1; 

else 

echo $pagecount; 

?> 

"class="a1"> 下 一 页 </a>&nbsp; 

<a href="<?php echo $_SERVER['PHP_SELF];?>?page=<?php echo $pagecount'?>” class="a1"> 尾 页 
</a></div> 

</td> 

</tr> 

</table> 


wk 


9.5 编辑 留 
9.5.1 编辑 留言 概述 
为 了 保证 留言 者 的 留言 信息 不 被 他 人 随意 修改 ， 留 言 用 户 只 能 对 个 人 的 留言 内 容 进 行 修改 ， 无 权 


修改 他 人 的 留言 内 容 。 编 辑 留 言 页 面 的 运行 结果 如 图 9.14 所 示 。 单 击 “ 编 辑 ” 超 链接 ， 打 开 编 辑 留 言 
窗口 ， 修 改 留言 内 容 后 单 击 “编辑 ”按钮 ， 即 可 完成 修改 留言 操作 。 


[Ea 
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图 9.14 编辑 留言 页 面 的 运行 结果 


9.5.2 ”编辑 留言 页 面 设计 
为 了 使 用 户 能 够 查看 到 原 留言 的 内 容 ， 在 设计 编辑 窗口 时 ， 采 用 一 个 弹出 窗口 作为 编辑 表单 。 纺 
辑 留言 页 面 的 设计 效果 如 图 9.15 所 示 。 


编辑 留言 


是 


EEC 


图 9.15 编辑 留言 页 面 的 设计 效果 
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编辑 留言 页 面 的 设计 流程 如 下 : 
(1) 用 户 登 录 后 , 系统 将 数据 库 中 所 保存 的 用 户 信息 与 用 户 的 登录 信息 进行 比较 , 如 果 二 者 相同 ， 
则 在 该 条 留言 后 显示 “编辑 ”按钮 。 代 码 如 下 : 


<?php 

$sqlu=mysql_query("select usernc from tb_user where id=".$info["userid"]."",$conn); 
$infou=mysql_fetch_array($sqlu); 

if($infou["usernc"]==$_SESSION["unc"]X 

?> 

<a href="javascript:openeditwindow(<?php echo $info[id];?>)" class="a1"> 编 辑 </a> 
<?php 

?> 


(2) 当 用 户 单 击 “ 编 辑 ” 按 钮 后 ， 将 通过 自 定义 的 openeditwindow() 方 法 打开 一 个 新 编辑 窗口 来 
编辑 用 户 留 言 。 代 码 如 下 : 


<script language="javascript"> 

function openeditwindow(xX{ /定义 openeditwindow() 方 法 打开 弹出 窗口 
window.open("editleaveword.php?id="+x,"newframe","top=100,left=200,width=450,height=280,menubar=no, 
location=no,scrollbars=no,status=no"); 

} 


</script> 


编辑 留言 页 面 涉 及 的 HTML 表单 的 重要 元 素 如 表 9.2 所 示 。 
表 9.2 编辑 留言 页 面 涉及 的 HTML 表单 的 重要 元 素 


名 称 类 型 含义 重要 属性 

name="forml" method="post" action="<?php echo $_ SERVER["PHP_ 

forml form 表单 . . 
SELF"]:?>" onSubmit="return chkinput(this)" 

i we 留言 主题 oe He type="text" class="inputcss" size="45" value="<?php echo 
Sinfo[title]:?>" 

content textarea 留言 内 容 name="content" cols="52" rows="12" class="inputcss" 

id hidden 留言 内 容 id type="hidden" name="id" value="<?php echo $_GET[id];?>" 

submit submit “编辑 ”按钮 type="submit" value=" 编 辑 " class="buttoncss" name="submit" 

reset reset “取消 ”按钮 type="reset" name="reset" value=" 取 消 " class="buttoncss" 


9.5.3 ”编辑 留言 内 容 功能 实现 


为 了 保证 用 户 的 留言 内 容 不 被 他 人 私自 更 改 ， 在 具体 实现 该 功能 时 ， 采 用 每 个 用 户 只 能 对 自己 的 
留言 内 容 进 行 更 改 的 方式 。 其 实现 原理 如 下 : 首先 ， 将 标识 用 户 身份 的 Session 变量 的 值 与 数据 库 中 该 
条 留言 对 应 的 留言 者 进行 比较 ， 如 果 二 者 相同 ， 则 说 明 该 条 留言 为 当前 登录 用 户 所 发 表 的 ， 在 该 条 留 


@ 
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言 后 显示 “编辑 ”按钮 。 当 用 户 单 击 “ 编 辑 ” 按 钮 后 ， 即 可 实现 对 留言 信息 的 更 改 ， 并 且 在 关闭 弹出 
窗口 的 瞬间 ， 留 言 信息 的 显示 页 面 也 会 自动 进行 刷新 。 
执行 编辑 留言 操作 的 editleaveword.php 文件 的 代码 如 下 : 


<?php 

include_once("conn.php"); 

$id=$_GET["id"]; 

/| 判断 是 否 单 击 “编辑 ”按钮 ， 如 果 是 执行 如 下 用 于 实现 用 户 留 言 信息 更 改 的 代码 ， 并 刷新 父 窗口 和 关闭 弹出 窗口 
if($_POST["submit"]!="){ 

if(mysql_query("update tb_leaveword set title=".$_POSTT["title"].",content=".$_POST["content"]." 
where id=".$_POST["id"]."",$conn)X{ 

echo "<script>alert(' 留 言 更改 成 功 ! ');window.opener.location.reload();window.close();</script>"; 
}else{ 

echo "<script>alert(' 留 言 更 改 失 败 ! ');history.back();</script>"; 

} 

exit; 

$sql=mysql_query("select * from tb leaveword where id=".$id."",$conn); 

// 如 果 用 户 未 单 击 “ 编 辑 ” 按 钮 ， 则 查询 该 用 户 的 留言 内 容 ， 并 最 终 将 未 编辑 的 留言 内 容 显 示 在 表单 中 
$info=mysql_fetch_array($sql); 


?> 
DV 
4 编辑 用 户 留 言 ， 用 户 实现 关闭 弹出 窗口 ， 自 动 刷 新 父 窗口 ， 是 通过 如 下 语句 实现 的 : 


window.opener.location.reload( ); 


上 述 代 码 的 实现 原理 是 : 在 用 window.close() 语 句 关闭 弹出 窗口 前 , 调用 父 窗 口 的 reload() 
方法 实现 父 窗口 的 刷新 。 


| 


9.6 删除 留 


9.6.1 删除 留言 概述 

如 果 用 户 发 表 非 法 留言 ,留言 本 管理 员 有 权 对 其 进行 删除 。 管理 员 登 录 后 将 在 每 条 留言 后 显示 “ 删 
除 ” 按 钮 ， 当 管理 员 单 击 该 按钮 后 ， 将 弹出 一 个 对 话 框 提 示 管 理 员 是 否 真正 删除 该 条 留言 。 如 果 单 击 
对 话 框 中 的 “确定 ”按钮 ， 则 该 条 留言 将 被 删除 ， 反 之 不 做 任何 操作 。 
说明 

Wo < 本 模块 的 管理 员 用 户 名 为 mr， 密 码 为 mrsoft。 


删除 留言 页 面 的 运行 结果 如 图 9.16 所 示 。 
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图 9.16 ”删除 留言 页 面 的 运行 结果 
9.6.2 ”删除 留言 页 面 设计 
删除 留言 页 面 的 设计 效果 如 图 9.17 所 示 。 


图 9.17 删除 留言 页 面 的 设计 效果 


判断 当前 登录 用 户 是 否 为 管理 员 ， 如 果 是 则 显示 “删除 ”按钮 ， 代 码 如 下 : 


<?php 

if($_SESSION["unc"]="™"){ // 判 断 用 户 是 否 登 录 

$sqld=mysql_query("select usertype from tb_user where usemc=".$_SESSION["unc"]."",$conn); 
$infod=mysql_fetch_array($sqld); 

if($infod["usertype"]==1X{ /| 判断 登录 用 户 是 否 为 管理 员 


?> 


@ 
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<a href="javascript'if(window.confirm(' 确 定 删除 该 留言 信息 吗 ? ')==true) 

08 {window.location.href='deleteleaveword.php?id=<?php echo $info[id];?>";}" class="a1"> 删 除 </a> 
// 如 果 是 则 显示 “删除 ”按钮 

<?php 


} 
?> 


9.6.3 ”删除 留言 内 容 功能 实现 


当 用 户 确认 删除 留言 后 ， 将 进入 留言 删除 处 理 页 deleteleaveword.php 文件 中 ， 根 据 超 链接 传递 的 
id 值 ， 执 行 删除 操作 。deleteleaveword.php 的 代码 如 下 : 


<?php 

include_once("conn.php"); // 包 含 数据 库 连接 文件 
if(mysql_query("delete from tb_leaveword where id=".$_GETI"id"]."",$conn)X{ 
echo "<script>alert(' 留 言 删除 成 功 ! ');history.back();</script>"; 


}else{ // 删 除 指定 的 留言 ， 并 给 出 提示 
echo "<script>alert(" 留 言 删除 失败 ! ');history.back();</script>"; 

了 

?> 


9.7 查询 留言 
9.7.1 查询 留言 概述 


随 着 留言 本 使 用 年 限 的 增多 ， 留 言 的 数量 会 越 来 越 多 。 为 了 方便 浏览 者 的 浏览 和 快速 定位 到 指定 
的 留言 ， 在 制作 留言 本 时 ， 设 计 了 可 以 按 留 言 主题 和 留言 内 容 进 行 模糊 查找 、 按 留言 者 进行 匹配 查找 


的 查询 留言 模块 。 
查询 留言 页 面 的 运行 结果 如 图 9.18 所 示 。 
有 旧 生 证 中 > 
-一 - 今天 是 : | 您 当前 的 位 置 : 明日 留言 丰 》》 查询 留言 


请 抽查 涛 广 式 : [请 允 委 到 理光 


A 祝 所 有 读者 用 友人 五 一 节 快 乐 ， 
Lg 
8910112131 
15 8 1 20 21 
IIHR 


访 伟 汀 显 Tt 直人 oicqa 


9.18 查询 留言 页 面 的 运行 结果 
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9.7.2 查询 留言 页 面 设计 


查询 留言 页 面 的 设计 效果 如 图 9.19 所 示 。 


图 9.19 查询 留言 页 面 的 设计 效果 
为 了 方便 用 户 ， 所 以 设计 了 可 以 按 多 种 方式 进行 查找 的 查询 表单 ， 代 码 如 下 : 


<div align="center"> 请 选择 查找 方式 : 

<select name="type"> 

<option value=""> 请 选择 </option> 

<option value="1"> 主 题 </option> 

<option value="2"> 内 容 </option> 

<option value="3"> 留 言 者 </option> 

</select>&nbsp;<input type="text" name="keyword" size="25" class="inputcss">&nbsp; 
<input type="submit" value=" 查 询 " class="buttoncss" name="submit"> 

</div> 


查询 留言 页 面 涉 及 的 HTML 表单 的 重要 元 素 如 表 9.3 所 示 。 
表 9.3 ”查询 留言 页 面 涉及 的 HTML 表单 的 重要 元 素 


名 称 重要 属性 

eit name="form1" method="post" action="<?php echo $_ SERVER["PHP_ 
SELF"];?> " onsubmit="return chkinput_search(this)" 

type name="type" 

keyword type="text" name="keyword" size="25" class="inputcss" 

submit type="submit" value=" 查 询 " class="buttoncss" name="submit" 


9.7.3 ”查询 留言 内 容 


当 用 户 按 要 求 添加 完 查 找 关 键 字 ， 选 择 了 查找 方式 后 ， 单 击 “ 查 询 ” 按 钮 ， 数 据 将 被 提交 到 本 页 ， 
在 本 页 完成 数据 的 查询 操作 ， 并 且 输 出 查询 结果 。searchword.php 的 关键 代码 如 下 : 


@ 
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<?php 

if($_POST["submit"]!="){ // 判 断 用 户 是 否 单 击 “ 查 询 ” 按 钮 ， 如 果 是 则 开始 查询 工作 
$type=$_POSTT["type"]; 

S$keyword=$_POST["keyword"]; 

if($type==1X{ /根据 用 户 提交 的 查询 类 型 定义 查询 语句 
$sql=mysql_query("select * from tb_leaveword where title like '%".$keyword."%"", $conn); 
}elseif($type==2)X{ 

$sql=mysql_query("select * from tb_leaveword where content like '%".$keyword."%", $conn); 
}elseif($type==3X{ 

$sql0=mysql_query("select id from tb_user where usemc=".$keyword."",$conn); 
$info0=mysql_fetch_array($sql0); 

$sql=mysql_query("select * from tb leaveword where userid=".$info0["id"]."",$conn); 

下 


$info=mysql_fetch_array($sql); 
if($info==falseX{ // 判 断 是 否 查找 到 相关 内 容 ， 如 果 没 查找 到 ， 则 显示 提示 信息 
echo "<br><br><div align=center> 对 不 起 ， 没 有 查找 到 您 要 查找 的 内 容 ! </div>"; 


dof /通过 循环 显示 查询 结果 


?> 

“…// 显 示 查 找 内 容 

<?php 
}while($info=mysql_fetch_array($sql)); 
} 


?> 


9.8 技术 提炼 


9.8.1 将 数据 保存 到 数据 库 中 


将 用 户 发 表 的 留言 保存 到 MySQL 数据 库 是 通过 SQL 语言 的 insert 语句 实现 的 。PHP 通过 
mysql_query() 函 数 向 MySQL 数据 库 中 发 送 SQL 命令 ， 所 以 mysql_query0 函 数 在 数据 库 编程 过 程 中 具 
有 极其 重要 的 作用 ， 下 面 将 对 该 函数 进行 介绍 。 

mysql_query() 函 数 用 于 向 与 指定 的 连接 标识 符 所 关联 的 数据 库 服务 器 中 发 送 一 条 查询 语句 。 其 语 
法 如 下 : 


resource mysql_query (string query [, resource link_identifier]) 

参数 query 为 必 选 参数 ， 用 于 指定 向 数据 库 中 发 送 的 查询 语句 ，link_identifier 为 可 选 参数 ， 是 
mysql_connect() 函 数 成 功 连接 数据 库 后 所 返回 的 连接 标识 , 如 果 省 略 了 该 参数 , 则 使 用 最 近 一 次 与 数据 
库 建立 连接 后 所 返回 的 连接 标识 。 
9.8.2 ”通过 JavaScript 实现 弹出 窗口 


通过 JavaScript 实现 弹出 窗口 ， 主 要 是 通过 调用 window 对 象 的 open0 方 法 实现 的 ， 并 将 实现 调用 
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的 代码 封装 到 一 个 单独 的 自 定义 方法 中 ， 最 后 通过 HTML 元 素 的 onclick 事件 对 该 方法 进行 调用 。 
open() 方 法 的 语法 如 下 : 


open("url","name", "features") 
该 方法 的 参数 说 明 如 表 9.4 所 示 。 
表 9.4 open() 方 法 的 参数 说 明 


参数 取 值 说 阴 
可 选 参数 。 用 于 指定 弹出 新 窗口 的 URL 地 址 ， 如 果 省 略 该 参数 或 其 值 是 一 个 空 字符 串 ， 那 么 
新 窗口 就 不 显示 任何 文档 
新 窗口 的 名 字 ， 如 果 该 名 字 是 一 个 已 经 存在 的 窗口 ， 那 么 open() 方 法 就 不 再 创建 一 个 新 窗口 ， 
> 而 只 是 返回 这 个 窗口 的 引用 ， 在 这 种 情况 下 参数 features 将 被 忽略 
人 主要 是 显示 窗口 的 一 些 特征 ， 该 参数 如 果 不 设 定 ， 则 显示 这 个 窗口 的 所 有 特征 。features 的 常 
用 特征 请 参见 表 9.5 
表 9.5 参数 features 的 常用 特征 
名 称 用 法 
channelmode 指定 窗口 是 否 应 该 以 频道 的 方式 显示 ， 取 值 有 yes/no 或 1/0 
fullscreen 指定 窗口 是 否 全 屏 显 示 ， 取 值 有 yes/no 或 /0 
menubar 指定 窗口 是 否 有 菜单 栏 ， 取 值 有 yes/no 或 1/0 
scrollbars 指定 窗口 是 否 有 水 平 及 竖 直 滚动 条 ， 取 值 有 yes/no 或 1/0 
status 指定 窗口 是 否 有 状态 栏 ， 取 值 有 yes/no 或 1/0 
toolbar 指定 窗口 是 否 有 工具 栏 ， 取 值 有 yes/no 或 1/0 
location 指定 窗口 是 否 有 地 址 栏 ， 取 值 有 yes/no 或 1/0 
directions, 指定 新 建 窗 口 是 否 有 标准 目录 按 扭 ， 取 值 有 yes/no 或 1/0 
Tesizable 指定 新 窗口 大 小 是 否 可 以 调整 ， 取 值 有 yes/no 或 1/0 
top 以 像素 为 单位 指定 新 窗口 上 沿 与 屏幕 上 沿 的 距离 
left 以 像素 为 单位 指定 新 窗口 左 沿 与 屏幕 左 沿 的 距离 
width 以 像素 为 单位 指定 新 窗口 的 宽度 
height 以 像素 为 单位 指定 新 窗口 的 高 度 


9.8.3 包含 文件 函数 


在 开发 Web 程序 时 ， 经 常会 在 不 同 的 页 面 中 实现 相同 的 功能 或 执行 相同 的 代码 。 为 了 提高 程序 开 
发 效率 ， 可 以 将 相同 的 代码 存储 到 单独 的 文件 中 ， 然 后 用 包含 语句 在 页 面 中 包含 该 文件 即 可 。 
include_once 语句 用 于 在 脚本 执行 期 间 包含 并 运行 指定 的 文件 。 其 语法 如 下 : 


include_once(filename) 


参数 filename 用 于 指定 要 包含 的 文件 。 


@ 
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9.8.4 MySQL 数据 库 的 函数 


PHP 中 提供 了 一 组 操作 MySQL 数据 库 的 函数 , 应 用 这 些 函数 可 以 方便 地 对 MySQL 数据 库 进 行 管理 。 

mysql_fetch_array() 函 数 获得 数据 库 中 满足 函数 mysql_query0 中 的 SQL 语句 的 记录 , 其 返回 值 是 一 
个 数组 ， 该 数组 的 下 标 可 以 是 字段 名 ， 也 可 以 是 索引 下 标 ， 数 组 元 素 的 值 是 某 个 字段 的 内 容 。 该 函数 
会 使 记录 指针 自动 向 下 移动 ， 当 移动 到 最 后 一 行将 返回 一 个 false 值 。 其 语法 如 下 : 

array mysql_fetch_array(int result,int[result_type]) 

该 函数 的 参数 说 明 如 表 9.6 所 示 。 

表 9.6 mysql_fetch_array() 函 数 的 参数 说 明 
说 ”了 明 

必 选 参数 。mysql_query() 函 数 向 MySQL 数据 库 发 送 查 询 命令 的 返回 标识 
可 选 参数 。 如 果 该 参数 取 MYSQL_BOTH， 则 该 函数 将 返回 一 个 同时 包含 关联 索引 和 数字 索引 的 数 


组 ; 如 果 该 参数 取 MYSQL_ ASSOC， 则 返回 一 个 关联 索引 的 数组 ， 如 果 该 参数 取 MYSQL_NUM， 
则 返回 一 个 数字 索引 的 数组 


参数 取 值 


result 


result_type 


/ 

人 培 轨 与 该 函数 功能 类 似 的 函数 还 有 mysql fetch_rows()、mysql_result()、mysql_fetch_object() 
等 。mysql_fetch_rowsO) 函 数 返 回 数 组 的 下 标 为 数值 索引 下 标 。 函 数 mysql_result() 有 两 个 参数 ， 其 中 
第 一 个 参数 也 是 mysql_query() 的 返回 结果 ， 而 第 二 个 参数 可 以 是 字段 的 偏 移 量 或 者 字段 名 ， 一 定 注 
意 它 返回 的 结果 不 是 数组 ， 而 是 MySQL 结果 集中 一 个 单元 的 内 容 。 函 数 mysql_fetch_object0 的 返 
回 结果 是 一 个 对 象 ， 使 用 时 只 能 通过 字段 名 来 返回 结果 。 


9.9 本 章 小 结 


本 章 主要 对 留言 本 的 设计 思路 和 业务 流程 进行 了 介绍 ， 并 详细 介绍 了 PHP 连接 和 管理 MySQL 数 
据 库 的 方法 及 留言 本 各 模块 的 实现 过 程 。 通 过 本 章 的 学 习 ， 不 仅 可 以 提高 读者 自身 的 基础 ， 而 且 可 以 
全 面 掌握 适合 于 中 小 型 企业 应 用 的 留言 本 的 制作 方法 ， 最 终 可 以 在 此 基础 上 进行 延伸 ， 制 作出 更 大 规 
模 的 Web 项 目 。 


UIE 


博客 模块 


(PHP+MySQL+mysqli 实现 ) 


如 果 有 人 问 “ 什 么 是 Web 2.0?“， 丽 怕 没 几 个 人 能 说 得 清楚 ， 但 
是 要 问 什么 是 博客 ， 哪 怕 是 从 不 上 网 的 人 也 是 耳熟能详 的 了 。 博 客 正 
是 Web 2.0 概念 中 重要 的 组 成 部 分 之 一 《大 家 熟知 的 还 包括 IM 即时 
通 和 R95 阅读 器 )。 

Blog (博客 )， 全 名 \Weblog， 后 来 缩写 为 Blog。Blogger 就 是 写 
Blog 的 人 ， 习 惯 于 在 网 上 写 出 上 日记、 发 布 个 人 照片 、 展 示 个 性 自我 的 
用 户 群 体 。 对 于 Blog/Blogger 的 中 文 名 称 ， 有 翻译 成 “博客 "， 也 有 
翻译 为 “网 志 ”， 但 大 多 数 人 都 已 经 认可 了 “博客 ”。 

通过 阅读 本 章 ， 可 以 学 习 到 : 


WI 系统 设计 思路 让 数据库 设 计 

MH ”博客 空间 个 人 首页 设计 WI 我 的 文章 模块 的 设计 
H 文章 浏览 模块 设计 H 文章 管理 模块 设计 
中 ”好友 管理 模块 设计 MW 小 纸 条 管理 模块 设计 


用户 管理 模块 设计 


第 10 章 博客 模块 (PHP+MySQL+mysqli 实现 ) 


10.1 博客 模块 概述 


10.1.1 ”模块 概述 


博客 ， 即 在 网 络 上 发 表 文 章 、 展 现 个 性 的 一 类 人 ， 而 博客 系统 ， 是 为 这 类 人 群 提供 一 个 相互 交流 、 学 
习 的 平台 。 博 客 的 两 大 基本 功能 是 共享 与 交流 。 共 享 ， 就 是 将 文章 、 图 片 、 心 得 等 一 些 在 传统 概念 中 属于 
私人 的 东西 ， 拿 出 来 和 多 数 人 一 起 分 享 ， 交流， 就 是 有 着 同样 兴趣 、 爱 好 、 语 言 的 一 类 人 之 间 的 联系 。 


10.1.2 功能 结构 


本 章 的 博客 系统 主要 分 为 个 人 博客 管理 和 博客 后 台 管 理 两 部 分 。 

个 人 博客 管理 ， 主 要 功能 包括 个 人 管理 、 文 章 管理 、 相 册 管 理 、 好 友 管理 、 留 言 管理 和 小 纸 条 。 
个 人 博客 管理 功能 结构 如 图 10.1 所 示 。 

博客 后 台 管理 ， 主 要 是 对 博客 用 户 和 管理 员 的 管理 。 博 客 后 台 管 理 包括 用 户 管理 、 文 章 管理 、 相 
册 管 理 、 管 理 员 管理 、 系 统 日 志和 数据 备份 。 博 客 后 台 管理 的 功能 结构 如 图 10.2 所 示 。 


个 人 博客 管理 
2 区 博客 后 台 管 理 
添 | | 查 确 各 查 | 出 | | 纸 上 本 
加 | | 看 认 | | 入 | 十 | 除 | | 条 复 和 
好 | | 好 添 | | 空 | | 名 | | 名 | | 到 纸 
友 加 | | 间 | | 言 | | 言 | | 表 条 
1 1 
不 人 管理 文章 管理 相册 管理 
查 添 | | 管 | | 攻 
看 加 | | 理 | | 除 
次 相 | | 相 
料 有 册 册 
添 | | 型 
加 | | 除 
相 | | 相 
册 | | 册 
图 10.1 个 人 博客 管理 功能 结构 图 图 10.2 博客 后 台 管理 功能 结构 图 


10.1.3 ”程序 预览 


为 了 让 读者 对 本 模块 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 模块 的 儿 个 页 面 运行 效果 图 ， 如 果 想 查 


_ 国 
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看 完整 的 效果 图 ， 请 参见 光盘 源 程序 。 
博客 首页 页 面 运行 效果 如 图 10.3 所 示 。 


i DB Bis 


博客 皇 帝 

my BLD6 的 博客 

工 DyE Y0U 的 博客 
格格 的 博客 


di Di 


博客 兵 帝 

my HLDc 的 博客 

工 LVE YO 的 博客 
5F 际 格格 的 博客 


0431-64946961 84978982 


图 10.3 博客 首页 页 面 运行 效果 
个 人 博客 管理 页 面 运行 效果 如 图 10.4 所 示 。 


志和 on 入 网 省 世界 ， 积 当天 天 快乐 
| CEL 
性 个 人 管理 | tre [ 3 | 上 人 图/ 0o¥ 
ad | 
总 文章 管 理 个 人 信息 (可 修 避 》 
各 相册 管理 
1 CE 
上 好 友 管 理 | | 
PR EE | a soro 禾 
只 ee 
ma | | 
个 人 主页 。 | tt/ | 
直言 | 


图 10.4 个 人 博客 管理 页 面 运行 效果 
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博客 用 户 管理 页 面 运行 效果 如 图 10.5 所 示 。 


陈竺 中 必 
请 本 月 户 耻 和 el 于。 生日 [2 
DD le 习 20u1-0L-0t o 理 六 结社 
口 2¢ ml 男 2011-01-01 o 否 访 结 详 经 秦 料 
口 bE Dll 男 LOLGL 5 再 诗 洁 详 讶 次 料 
口 2 证 际 格格 T784528q4 com 王仁 丽 总 2011-01-01 3 要 笠 洁 详 9ER 料 
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图 10.5 博客 用 户 管理 页 面 运行 效果 
图 片 管理 页 面 运行 效果 如 图 10.6 所 示 。 


上 传 用 户 : 旗 联 格 客 
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图 10.6 图 片 管理 页 面 运行 效果 
10.2 ”数据 库 设计 


本 模块 属于 中 小 型 个 人 网 站 ， 毫 无 争议 ， 本 模块 采用 的 依然 是 PHP+MySQL 这 对 黄金 组 合 ， 无 论 


PHP 典型 模块 开发 全 程 实录 
是 从 成 本 、 性 能 、 安 全 上 考虑 ， 还 是 从 易 操作 性 上 考虑 ，MySQL 都 是 最 佳 选择 。 
10.2.1 创建 数据 库 与 数据 表 
博客 系统 使 用 的 数据 库 是 db_blog, 在 数据 库 中 一 共 使 用 了 9 个 数据 表 。 各 个 表 的 名 称 和 作用 如 图 107 


所 示 。 


胃 服务 器 : localhost 名 数据 库 : db_blog 
表 氢 作 记录 教 外 ” 类 型 整理 说 明 


th_admin 加 加 天 国 X 1 MYSAM gb2312_chinese_ci 。 管理 员 信息 表 
tb_article [= El 10 NyISAM gb2312_chinese_c 。 博客 文章 列表 
th_ird 军国 比 便 xX 1 WySAM ”gb2312_chinese_d 。 用户 好 友信 息 志 
tb_log 宙 加 线 国 Xx 127 MYSAM ”9b2312_chinese_d 。。 文章 日 志 信 息 表 
th_member 世 回 大 面 Xx 12 MySAM ”9b2312_chinese_c ”用 户 信息 数据 表 
th_mess 雷 回 曾 X 7 MYSAM gb2312_chinese_d 。 用 户 留 计数 据 表 
tb_review 加 图 立轴 X 31 MYSAM ”9b2312_chinese_g 。 文章 评论 表 
tb_script 站 回 线 硬 X 13 MYSAM ”9b2312_chinese_q 。 小 纸 条 列表 
th_uppics 站 加 大国 Xx 4 MSAM 。 gb2312_chinese_c 。 上 传 图 片 信息 
9 个 表 和 总计 216 MyISAM gb2312_chinese_ci 


图 10.7 数据 表 列 表 
下 面 来 具体 看 一 下 其 中 几 张 重要 数据 表 的 结构 设计 。 
1. tb_article (博客 文章 列表 ) 
tb_article 表 包 含 类 别名 称 、 文 章 标题 、 文 章 内 容 等 字段 。tb_article 表 结 构 及 说 明 如 图 10.8 所 示 。 


加 服务 器 : localhost ， 邦 数据 库 : db_blog ， 四 表 : tb_article 
亲 浏 览 ， 架 结构 “加 SQL 万 搜 索 肝 白 入 ”了 V 叶 出 ” 较 Import 区 操作 而 清空 四 到 除 

字段 类 型 整理 属性 Null 黑 认 颜 外 说 明 
ru int(q) 否 auto_increment 自动 号 
厂 typename varchar(20) gb2312.chinese-ci 否 类 别名 称 
mtitle varchar(50) 。 gb2312-chinese-_ci 否 文章 标题 
content mediumtext gb2312-chinese_cl 否 文章 内 容 
厂 author varchar(50) gb2312_chinese_ci 否 文章 作者 
厂 firsttime timestamp ONUPDATE CURBENT. TuEsTAuP 否 CURRENT_TIMESTAMP 发 表 时 间 
厂 artquote varchar(300) gb2312-chinese_ci 否 是 否 引用 
hitnum int(4) i 订 点 击 率 
厂 renum ine(q) 二 局 古 回复 率 
厂 examine Int0) 百 0 审核 
厂 isnominare inr0) 否 0 是 否 推荐 


图 10.8 tb_article 表 结 构 及 说 明 
2. tb_frd (用 户 好 友信 息 表 ) 
tb_frd 表 包 含 好 友 名 称 、 添 加 人 、 好 友 等 级 和 添加 时 间 等 字段 。tb_frd 表 结 构 及 说 明 如 图 10.9 所 示 。 
3. tb_member (用 户 信息 数据 表 ) 


tb_member 表 包 含 用 户 的 基本 资料 ， 如 E-mail、 真实 姓名 和 出 生日 期 等 , 还 有 个 人 博客 中 的 一 些 参 
数 ， 如 点 击 率 、 文 章 类 别 、 图 片 类 别 、 是 否 推荐 等 。tb_member 表 结 构 及 说 明 如 图 10.10 所 示 。 


他 
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中 服务 器 : localhost 》 厚 数据 库 : db_blog ， 国 表 : rb_frd 
| 硬 剂 览 ”连结 构 如 sQL 万 搜 索 了: 医 入 陋 导 出 阐 Import 贫 扣 作伪 清空 “器 副 除 
字段 类 型 整理 属性 Null 黑 认 额外 说 明 

过 int(4) 否 auto_increment 自动 篇 号 
frdname varchar(100) gb2312_chinese_ci 否 好 友 名 称 
frdmem varchar(l00) gb2312_chinese_ci 否 添加 入 
frdlevel int(1) 否 好 友 等 级 
addtime timestamp ONUPDATE CURRENT_TIUESTAWP 否 CURRENT_TIMESTAMP 添加 时 间 


0.9 tb _frd 表 结构 及 说 明 


加 服务 器 : localhost ， 大 数据 库 : db_blog 国 表 : tb_member 
周济 览 ” 阿 结构 “加 SQL 万 朱 索 于 插入。 此 导出 

字段 类 型 整理 额外 说 明 
ud int(4) 否 auto_increment “自动 编号 
name varchar(50) ”gb2312-chinese-ci 否 用 户 名 称 
pwd varchar(50) ”gb2312-chinese-ci 否 用 户 密码 
question varchar(50) gb2312_chinese_cl 否 密 保 问题 
answer varchar(50) gb2312_chinese_ci 1 密 保 答案 
email varchar(50) gb2312_chinese_cl 否 Enail 
realname varchar(50) gb2312_chinese_ci 否 真实 姓名 
Sex char0) 9b2312-chinese-cl 否 性 别 
birthday dare 否 2008-06-12 出 生日 其 
tel varchar(20) 。 gb2312_chinese_ci 否 电话 
address Varchar(200) gb2312-chinese-ci 否 地 址 
homepage varchar(200) gb2312_chinese_cl 否 个 人 主页 
qq varchar(l0) gb2312_chinese_ci 否 aq 号 码 
unwrite varchar(200) gb2312-chinese-cl 否 个 人 签名 
headgif varchar(200) gb2312-chinese_ci 否 头像 
hitnum int(4) 否 0 点 击 率 
upfile Int(4) 否 0 上 传 文件 
uppics int(4) 否 0 上 传 图 片 
lasttime timestamp ONUPDATE CURRENT_TIVESTAWP 否 CURRENT_TIMESTAMP 最 后 登录 
freeze int0) 否 0 是 否 闵 结 
blogname varchar(100) gb2312_chinese_ci 否 博客 名 称 
blogurl varchar(200) gb2312_chinese_ci 否 博客 链接 
arttype varchar(300) gb2312_chinese_ci 否 文章 类 别 
pictype varchar(300) gb2312_chinese_ci 否 图 片 类 别 
isnew int0) 否 0 是 否 新 注册 
isnominate int0) 否 0 是 否 推荐 


图 10.10 tb_member 表 结 构 及 说 明 
4. tb_review 表 (文章 评论 表 ) 
tb_review 表 包 含 被 评论 的 文章 id、 评 论 内 容 、 评 论 人 以 及 评论 时 间 等 字段 。tb_review 表 结 构 及 说 
明 如 图 10.11 所 示 。 
5. tb_uppics 表 (上传 图 片 信息 ) 


tb_uppics 表 包 含 上 传 图 片 的 信息 ， 如 图 片 名 称 、 上 传 用 户 、 图 片 类 别 等 字段 。 tb_uppics 表 结 构 及 
说 明 如 图 10.12 所 示 。 
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加 服务 器 : localhost ， 辑 数据 库 - db_blog ， 国 表 : tb_review 
浏览 ” 阿 结 构 “ 毗 sQL 万 搜 索 了 : 括 入 此 导出 。 阿 Imporr。 做 哲 作 们 清空“ 器 删 除 


字段 类 型 整理 属性 Null 玛 认 额外 说 明 
id int(4) 否 auto_increment 。 自动 编号 
artid int(9) 否 文章 id 
content varchar(300) gb2312_chinese_ci 否 评论 内 容 
man varchar(100) gb2312_chinese_ci 否 评论 人 
firsttime timestamp ONUPDATE CURRENT TuESTAMP 否 CURRENT_TIMESTAMP i 


10.11 tb_review 表 结 构 及 说 明 


园 服务 器 : localhost 》 意 数据 库 : db_blog 》 国 表 : tb_uppics 
| 国 浏 览 。 队 结构 “, 观 SQL 户 搜 索 3 新 入 ”并 导 出 ” 圈 Import 侨 操 作 。 面 清 空 “ 涡 副 除 

字段 类 型 整理 属性 Null 黑 认 颜 外 说 明 
过 Int(4) 否 auto_increment ”自动 编号 
picname varchar(50) gb2312_chinese_cl 否 图 片 名 称 
picpath 。 varchar(100) gb2312-chinese_cl 否 图 片 路 径 
upauthor varchar(100) gb2312_chinese_cl 否 上 传 用 户 
pictype varchar(50) gb2312_chinese_cl 否 图 片 类 别 
hitnum Int(4) 否 0 点 击 率 
examine Inc0) 雪 是 否 审核 
uptime timestamp ONUPDATE CURRENT_TIMESTAMP 否 CURRENT_TIMESTAMP 上 传 时 间 


图 10.12 tb_uppics 表 结 构 及 说 明 


此 外 ， 还 有 管理 员 信 息 表 、 文 章 日 志 信 息 表 、 用 户 留言 数据 表 和 小 纸 条 列表 。 限 于 篇 幅 ， 这 里 不 


再 介绍 ， 


10:2.2 


读者 可 参见 本 书 附 赠 光 盘 中 的 数据 库 文件 。 
数据 库 操作 类 


为 了 方便 使 用 ， 减 少 代码 ， 本 模块 使 用 mysqli 扩展 存 取 数据 ， 与 mysql 扩展 相 比 ，mysqli 扩展 在 
以 下 方面 有 了 明显 的 提高 。 


加 回 


兼容 性 与 维护 性 : mysqli 扩展 可 以 很 容易 地 使 用 mysql 的 新 功能 ， 所 以 mysqli 拥有 与 mysql 
更 高 的 兼容 性 。 即 使 mysql 的 新 版 本 又 出 现 了 更 多 功能 ，mysqli 扩展 也 可 以 很 容易 地 支持 。 
面向 对 象 : mysqli 扩展 已 封装 到 一 个 类 中 ， 从 而 可 使 用 面向 对 象 的 方式 编程 。 即 使 对 面向 对 
象 不 了 解 ，mysqli 也 可 以 应 用 面向 过 程 的 编程 方式 来 实现 。 

速度 和 安全 性 : mysqli 扩展 执行 的 速度 比 之 前 版 本 的 mysql 扩展 快 了 很 多 。mysqli 扩展 支持 
mysql 新 版 本 的 密码 杂凑 (Passwoed Hashes) 和 验证 程序 ， 从 而 提高 了 应 用 程序 的 安全 性 。 
预 准备 语句 : 可 提高 重复 使 用 语句 的 性 能 ，mysqli 扩展 提供 了 对 预 准 备 语句 的 支持 。 

调试 功能 : mysqli 扩展 进一步 改进 了 调试 功能 ， 提 高 了 开发 效率 。 要 在 PHP 中 使 用 mysqli 扩 
展 , 需要 在 配置 文件 php.ini 中 添加 “extension=php_mysqli.dll” 设 置 ， 如 果 配 置 文 件 中 己 有 上 
述 设置 ， 确 保 extension 前 面 没有 “:”， 和 否则 将 其 去 掉 。 


下 面 介 绍 如 何 使 用 mysqli 扩展 来 存 取 数 据 库 。mysqli 提供 面向 对 象 和 面向 过 程 两 种 方式 来 与 数据 


库 交 互 。 
交互 。 


@ 


在 本 模块 中 ， 后 台 管 理应 用 面向 对 象 方式 与 数据 库 交 互 ， 前 台 应 用 面向 过 程 方式 与 数据 库 
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下 面 是 后 台 管 理 mysqli 扩展 应 用 的 面向 对 象 操作 类 。 在 这 里 连接 数据 库 就 是 实例 化 mysqli 系统 内 
定 的 一 个 类 ， 将 所 得 到 的 返回 集 “S$this->conn” 以 “S$this->result = $this->conn->query($sql)” 这 种 形式 
写 入 ， 其 存储 于 admin/conn/conn.php 文件 中 。 代 码 如 下 : 


// 定 义 私有 属性 ， 如 主机 名 、 用 户 名 、 用 户 密码 、 数 据 库 名 等 


<?php 
class opmysqli{ 


private $host = '127.0.0.1" /服务 器 地 址 
private $name = 'root'; /登录 账号 
private $pwd = 111'; /登录 密码 
private $dBase = 'db_blog'; /数据 库 名 称 
private $conn ="; /| 数据库 链接 资源 
private $result ="; /| 结果 集 
private $msg = "; /1 返回 结果 
private $fields; // 返 回 字段 
private $fieldsNum = 0; /| 返回 字段 数 
private $rowsNum = 0; /| 返回 结果 数 
private $filesArray = array(); 1// 返 回 字 段 数 组 
private $rowsArray = array(); /返回 结果 数组 


// 使 用 __construct 关键 字 初 始 化 类 。 这 与 mysql 扩展 写法 一 样 
function __construct($host=",$name=",$pwd=",$dBase="}{ 
if($host !=") 

S$this->host = $host; 

if($name != ") 

S$this->name = $name; 

if($pwd {= ") 

S$this->pwd = $pwd; 

if($dBase !=") 

S$this->dBase = $dBase; 

S$this->init_conn(); 


} 


连接 数据 库 需 要 实例 化 mysqli 类 。 该 类 是 PHP 环境 自 带 的 ， 所 以 不 要 管 它 从 哪里 来 ， 只 需 实例 化 
就 可 以 。 


function init_conn(){ 

$this->conn=new mysqli($this->host,$this->name, $this->pwd, $this->dBase); 
S$this->conn->query("set names gb2312"); 

} 


下 面 是 一 些 具体 的 操作 方法 ， 包 括 查询 结果 、 取 得 字段 数 、 更 新 、 删 除 、 添 加 记录 数 等 。 其 部 分 
代码 如 下 : 


/查询 结果 

function mysqli_query_rst($sql{ 
if($this->conn == "}{ 
$this->init_conn(); 


>) 
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} 
S$this->result = @$this->conn->query($sq)l); 


// 取 得 字段 数 

function getFieldsNum($sql){ 

$this->mysqli_query_rst($sql); 

S$this->fieldsNum = @S$this->result->num_fields($this->result); 


} 

// 取 得 查询 结果 数 

function getRowsNum($sql){ 
$this->mysqli_query_rst($sql); 
$this->rowsNum = @$this->result->num_rows; 
return $this->rowsNum; 


} 

// 取 得 字段 数 

function getRowsArray($sql){ 

S$this->mysqli_query_rst($sql); 

while($row = @$this->result->fetch_array(MYSQLI_ASSOC)){ 
S$this->rowsArray[] = $row; 

} 


return $this->rowsArray; 


. 

// 更 新 、 删 除 、 添 加 记录 数 

function uidRst($sql}{ 

if($this->conn == "){ 

$this->init_conn(); 

} 

$this->conn->query($sql); 

$this->rowsNum = @$this->conn->affected_rows; 
return $this->rowsNum; 


} 
// 最 后 实例 化 这 个 类 即 可 
$conne = new opmysqli(); 


前 台 使 用 mysqli 面向 过 程 与 数据 库 进行 交互 。 它 的 使 用 方法 类 似 于 mysql 扩展 。 例 如 , mysql_query0 
这 个 mysql 扩展 函数 ， 在 mysqli 扩展 中 是 写 为 mysqli_query()。 返 回 结果 集 在 mysql 中 是 mysql_query 
($sql，$conm)， 而 在 mysqli 扩展 中 则 是 mysqli_query($conn，$sqD)。 下 面 给 出 mysqli 面向 过 程 操 作 数据 
库 的 部 分 代码 ， 其 存储 于 center/conn/conn.php 文件 中 。 


// 连 接 数据 库 

function init_conn(){ 
$this->conn=mysqli_connect($this->host,$this->name,$this->pwd,$this->dBase); 
mysqli_query($this->conn,"set names gb2312"); 


和 
/查询 结果 
function mysqli_query_rst($sql){ 


@ 
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if($this->conn ==" 
S$this->init_conn(); 


} 
S$this->result = @mysqli_query($this->conn,$sql); 


} 

// 取 得 字段 数 

function getFieldsNum($sql){ 
$this->mysqli_query_rst($sql); 

S$this->fieldsNum = @mysqli_num_fields($this->result); 
: 


有 关 数 据 库 操作 类 的 讲解 完毕 ， 其 具体 代码 请 参考 本 书 光盘 中 的 内 容 。 
10.3 博客 空间 个 人 首页 


10.3.1 博客 空间 个 人 首页 概述 


博客 系统 的 模块 较 多 ， 限 于 篇 幅 ， 这 里 只 对 重点 模块 进行 讲解 。 首 先 来 看 个 人 首页 。 个 人 首页 由 
多 个 页 面 组 成 ， 例 如 热门 文章 、 文 章 分 类 、 留 言 列表 等 。 其 页 面 的 设计 结构 如 图 10.13 所 示 。 


登录 注册 ( 进入 后 台 馆 出 登录 ) 
Banner 
显示 新 消息 。 | 导航 栏 ( 我 的 文章 我 的 相册 我 的 资料 我 的 好 友 ) 
热门 文章 列表 
博客 信息 〔《 hotatphp) 
( personinfo php) 
文章 类 别 列表 
Fhp) 
晶 历 arttype -Php 
C date html ) 内 容 显示 区 
(默认 newatl php) 
图 片 类 别 列表 
Cpictype php ) 
留言 列表 
( messl php) 好 友 链 接 
Callfrd php) 
留言 板 ( lyb .php) 


10.13 博客 空间 个 人 首页 结构 及 布局 
首页 的 运行 效果 如 图 10.14 所 示 。 
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10.14 ”博客 首页 运行 效果 
10.3.2 首页 模块 实现 过 程 


个 人 首页 设计 极其 简单 , 总 体 是 采用 div+css 样式 设计 的 , 例如 导航 条 、 个 人 首页 的 头 部 、 外 层 等 ， 
都 是 采用 div+css 设计 的 。 然 后 再 使 用 包含 文件 结合 技术 , 把 所 有 需要 显示 在 首页 的 文件 和 配置 文件 引 
入 进来 即 可 。 有 具体 步骤 如 下 : 

(1) 在 centercenter.php 文件 中 加 载 各 种 文件 ， 例 如 配置 文件 、 数 据 库 类 文件 、 计 数 统计 文件 、 
CSS 样式 文件 、js 脚本 文件 等 。 代 码 如 下 : 


<?php 

session_start(); /开启 Session 
header('Content-Type:text/html;charset=gb2312'); // 设 置 页 面 编 码 
include_once '../config.php'; 1/ 加载 配置 文件 
include_once 'conn/conn.php'; /1 加载 数据 库 类 
include_once 'inc/count.php'; 1/ 加载 计数 器 
3 


<link href="../css/style.css" rel="stylesheet" type="text/css" /> 
<script language="javascript" src="../js/xmlhttprequest.js"></script> 
<script language="javascript" src="../js/center.js"></script> 

<script language="javascript" src="../js/date.js"></script> 

<script language="javascript" src="../js/personinfo.js"></script> 


(2) 创建 “注册 登录 ” 超 链接 。 判 断 当前 的 Session 值 是 否 为 室 ， 如 果 为 空 ， 就 显示 “登录 注册 ” 
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按钮 ， 否则， 显示 光临 信息 和 “进入 管理 页 面 ”的 超 链接 。 代 码 如 下 : 


<?php 

if(lempty($_SESSION[name']) and !is_nulll($_ SESSION[name]) 

?> 

您 好 ， 欢 迎 光临 <?php echo $_SESSION[name']; ?> 的 博客 ! 

<a id='blogmain' onclick="javascript:open('manage.php',”parent"," ,false)"> 进 入 管理 页 面 </a>&nbsp; 
<a onclick="javascript:open(logout.php'”parent,",false)"> 退 出 登录 </a>&nbsp; 

<?php 

}else{ 

?> 

<a onclick="javascript:window.open('login.php','login','width=300,height=200',false)"> 登 录 </a> 
<a onclick="javascript:window.open('register.php','register',",false)"> 注 册 </a>&nbsp; 

<?php 

} 

?> 


(3) 创建 Banner， 并 且 输 出 博客 空间 的 名 称 。Banner 下 面 的 一 行 ， 由 两 部 分 构成 。 一 部 分 是 信息 
消息 提醒 ， 另 一 部 分 是 导航 栏 。 新 消息 提醒 每 隔 一 段 时 间 ， 查 看 一 下 个 人 的 纸 条 信息 ， 如 果 发 现 新 纸 
FE ， 会 马上 显示 出 来 。 代 码 如 下 : 


<div id="nav"> 

<div id="newscrip">&nbsp; 

<Script>setlnterval("shownewmess()",10000);</script> 

</div> 

<div id="rightfloat"> 

<a onclick='javascript:open("center.php?uid=<?php echo $_GET[uid] ?>",” parent","",false) > 我 的 文章 </a> 

| <a onclick='avascriptopen("center.php?act=pics&uid=<?php echo $_GET[uid] ?>","_parent","",false)'> 我 的 相 
册 </a> 

| <a onclick='javascript:open("center.php?act=person&uid=<?php echo $_GET[uid] ?>","_parent","",false)'> 我 
的 资料 </a> 

| <a onclick='javascript:open("center.php?act=frd&uid=<?php echo $_GET[uid] ?>",” parent",",false)'> 我 的 好 
友 </a></div> 

</div> 


(4) 加 载 各 个 页 面 。 其中， 内容 显 示 区 是 首页 面 的 核心 ， 儿 乎 所 有 博客 空间 的 功能 都 将 在 内 容 显 
示 区 显示 。 内 容 显 示 区 的 代码 如 下 : 


<?php 

if(isset($_GET[act]) and $_GET['act] == "多 
include_once "newart1.php'; 

}else if(isset($_GET['act]) and $_GET[act] == 'see'X{ 
include_once 'showarticle1.php'; 

}else if(isset($_GET[act]) and $_GET['act] == 'pics'X{ 
include_once 'pics1.php'; 
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}else iflisset($_GET[act]) and $_GET[act] =='person')f 
include_once 'person.php'; 

}else if(isset($_GET['act]) and $_ GETTact] == 'frd'X{ 
include_once 'frd1.php'; 

}else if(isset($_GET['act]) and $_GET[act] == 'modart'X{ 
include_once "modart.php'; 

}else{ 

include_once 'newart1.php'; 


} 
?> 
</div> 


10.4 我 的 文章 


10.4.1 我 的 文章 概述 


进入 博客 个 人 空间 ， 或 者 单 击 博客 个 人 空间 的 “我 的 文章 ” 超 链 接 ， 将 在 内 容 显示 区 显示 博客 中 
已 存在 的 文章 的 简要 信息 列表 ， 如 图 10.15 所 示 。 其 中 ,包括 文章 标题 、 发 表 时 间 、 部 分 文章 内 容 和 一 
个 “阅读 全 文 ” 超 链 接 。 


音标 题 : 公司 标语 发 布 时 间 


-实现 数字化 的 次 达 ， 迎接 中 国 数字 出 版 的 腾飞 。 1 字 化 出 版 ， R 杰 你 和 
活 。? 让 用 户 花 更 少 的 钱 ， 获 得 更 多 、 更 专业 、 从 的 的 区 和 各 和 ?为 用 户 
创造 价值 ， 就 是 为 自己 创造 品牌 。? 让 忧 秀成 为 习惯 。? 坚 韦 、 创 新 、…… 


| 阅读 全 文 | 


文章 标题 : 公司 概况 发 布 时 间 


吉林 省 明日 科技 有 限 公 mp 
多 年 来 始终 致力 于 行业 管理 软件 、 数 字 化 域 的 实践 ， 目 前 已 和 多 家 国 
内 上 市 企业 形成 产品 合作 关系 ， 机 es 


| 阅读 全 文 | 


图 10.15 文章 列表 的 运行 结果 


单 击 “ 阅 读 全 文 ” 超 链接 ， 打 开 showarticlel.php 页 面 ， 将 显示 要 查看 的 文章 的 全 文 和 评论 信息 
网 还 显示 相关 的 一 些 操 作 ， 例 如 引用 文章 和 发 表 评论 。 如 果 是 本 人 的 博客 ， 登录 后 还 可 以 修改 六 
章 、 删 除 文章 和 删除 评论 ， 如 图 10.16 所 示 。 
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图 10.16 ”浏览 文章 页 面 的 运行 结果 


10.4.2 我 的 文章 功能 实现 过 程 


创建 newartl.php 文件 ， 完 成 我 的 文章 模块 设计 。 首 先 设置 分 页 信息 ， 然 后 生成 SQL 查询 语句 ， 
用 来 查找 当前 用 户 所 发 表 过 的 文章 ， 并 返回 查询 结果 。 实 现代 码 如 下 : 
<?php 


include_once 'conn/conn.php'; 
if(isset($_GET['uid])X 


Snum =7; // 每 页 显示 记录 数 
if(lisset($_GET['curpage']) 

$curpage = 1; 

Jelse{ 

S$curpage = $_GET['curpage]; /当前 页 

} 

$sql = "select * from tb_article where author = ".$_GET[uid].” order by id desc "; 
S$totnum = $conne->getRowsNum($sql); /记录 数 
Stotpage = ceil($totnum / $num); /页 数 

S$tmpsql = $sql." limit ".($num *($curpage-1)).",".$num; /SQL 查询 语句 
$arr = $conne->getRowsArray($tmpsql); /| 获取 数组 
$conne->close_rst(); 

?> 


如 果 获 取 到 的 数组 不 为 室 ， 那 么 循环 判断 是 否 为 引用 ， 如 果 不 是 ， 则 直接 输出 文章 信息 ; 如 果 是 ， 那 
么 根据 引用 的 文章 id 号 来 查找 原文 ， 并 输出 文章 信息 。 最 后 ， 添 加 一 个 “阅读 全 文 ” 超 链接 。 代 码 如 下 : 


区 
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if($arr!= "){ 

foreach($arr as $value}{ 

if(lempty($value['artquote]) and !is_null($value[artquote"])X{ 
$tmparr = explode(',',$value['artquote']); 

$sql1 = "select * from tb_article where id = ".$tmparr{0]; 
$artarr1 = $conne->getRowsArray($sql1); 

Sa rst(); 


jelsel 


} 
3; 


10.4.3 ”文章 浏览 功能 实现 过 


文章 浏览 页 〈showarticle1.php) 最 核心 的 功能 当然 是 浏览 文章 。 首 先 ， 根 据 超 链接 传递 的 文章 id， 
将 文章 信息 全 部 显示 出 来 。 代 码 如 下 : 


<?php 

include_once '../config.php'; /1/ 载 入 配置 文件 
include_once 'conm/conn.php'; // 载 入 数据 库 类 
$showartsql = "select * from tb_article where id = ".$_GET[artid]; /生成 查询 语句 
$artarr = $conne->getRowsArray($showartsql); // 返 回 数组 形式 的 记录 
$conne->close_rst(); /清空 结果 集 
if(lempty($artarr[OJ['artquote]) and !is_null($artarr[OJ[artquote])X ” // 判 断 文章 是 否 为 引用 
$tmparr = explode(",",$artarr[O]['artquote']); // 得 到 引用 的 来 源 地 址 
$sql1 = "select * from tb_article where id = ".$tmparrf0]; /根据 地 址 查询 数据 
$artarr1 = $conne->getRowsArray($sql1); // 返 回 结果 集 
$conne->close_rst(); 

}else{ 

} 

?> 


然后 ,实现 文章 引用 (其 实质 是 引用 文章 地 址 ) 。 当 其 他 的 博客 用 户 看 到 好 文章 时 ， 可 以 通过 “ 引 
用 ” 超 链接 ， 将 文章 id 保存 到 自己 的 文章 列表 中 ， 通 过 对 文章 id 的 访问 ， 可 以 省 去 手工 输入 的 麻烦 ， 
也 在 一 定 程度 上 节省 了 表 空 间 。 此 外 ， 如 果 博 主 想 对 自己 的 这 篇 文章 进行 宣传 ， 也 可 以 将 引用 地 址 复 
制 下 来 ， 粘 贴 到 其 他 地 方 来 提高 博客 空间 的 人 气 。 实 现 引用 的 代码 如 下 : 

[<a onclick="return artquote('<?php echo $_GET[uid] ?>''<?php echo $_GETTartid]; ?>')" style="color: 

#FF6600;"> 引 用 </a>]&nbsp; 来 自动 添加 到 您 的 博客 空间 <br /> 

<input id="copyadd" onclick="copyaddress()" type="text" size="50" value="<?php echo 'http://.$_SERVER[HTTP_ 

HOST].$_SERVER[REQUEST_URI]; ?>" readonly="readonly" style="border: 1px #FFFFFF solid;” /> 


该 段 代码 主要 通过 两 个 JavaScript 自 定义 函数 来 控制 。 第 1 个 是 artquote() 函 数 ， 将 博客 用 户 、 文 
章 id 传 给 showarticle_chk.php 页 ， 根 据 返 回 结果 进行 处 理 ; 第 2 个 是 copyaddress0 函 数 ， 主 要 实现 复 
制 当 前 页 面 URL 地 址 的 功能 。 脚 本 文件 js\centerjs 的 代码 如 下 : 


他 
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/引用 

function artquote(uid,quoteidX{ 
xmlhttp.open(get,'showarticle_chk.php?act=quote&uid='+uid+'&quoteid='+quoteid,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readystate == 4 && xmlhttp.status == 200X{ 
msg = xmlhttp.responseText; 

if(msg == '1){ 

alert(' 引 用 成 功 '); 

}else if(msg == '2){ 

alert(' 您 还 没有 登录 '); 

}else if(msg == '3){ 

alert(' 引 用 失败 '); 

}else if(msg == '4){ 

alert(' 不 允许 引用 自己 的 文章 '); 


Jelse{ 
alert(msg); 
} 
} 
xmlhttp.send(null); 
} 
// 自 动 复制 网 址 
function copyaddress(X{ 

S$('copyadd').select(); 

document.execCommand('COPY'); 

alert(' 成 功 复制 本 地 址 '); 
} 
showarticle_chk.php 页 处 理 和 文章 相关 的 操作 ， 包 括 添加 评论 、 删 除 评论 和 引用 。 主 要 代码 如 下 : 
<?php 
session_start(); /开启 Session 
header('Content-Type:text/html;charset=gb2312'); // 设 置 编码 
include_once 'conm/conn.php'; // 载 入 数据 库 类 
$artid = $_GET[artid]]; // 文 章 id 
$quoteid = $_GET[quoteid]; 11 引 用 id 
$name = $_SESSION[name'; // 当 前 用 户 
$uid = $_GET[uid]; // 原 文章 用 户 
$act = $_GET[act]; // 何 种 操作 
$reback = '0'; 
if(empty($act)X 


if($name != $uid){ 

$selectsql = "select renum from tb_article where id = ".$artid; 

$renum = $conne->getFields($selectsql,0); 

$upsql = "update tb_article set renum = ".(++$renum)." where id = ".$artid; 
$num = $conne->uidRst($upsql); 

} 

$content = $_GET[cont]; 

$sql = "insert into tb_review (artid,content,man) values(".$artid.",".$content.",".($name == "?' 匿 名 ':$name).")"; 
$num = $conne->uidRst($sql); 

$reback = "1"; 

}else if($act == 'quote'){ 1/ 如果 是 引用 操作 
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if(lempty($name) and !is_null($name)){f 

if($_GET[uid] == $namej{ // 如 果 是 引用 了 自己 的 文章 
S$reback = "4'; 

}else{ 

$chkquote = $conne->getRowsNum("select artquote from tb_article where author = ".$name.""); 
if($chkquote >= 1){ 

S$reback = '5"; 

}else{ 

$selectsql = "select renum from tb_article where id = ".$quoteid; 

$renum = $conne->getFields($selectsql,0); 

$upsql = "update tb_article set renum = ".++$renum." where id = ".$quoteid; 

$num = $conne->uidRst($upsql); 

$oldid = $conne->getFields("select artquote from tb_article where id = ".$quoteid, 0); 
if(lempty($oldid) and !is_null($oldid)X 

$saveid = $oldid….$quoteid; 

}else{ 

$saveid = $quoteid; 


} 

$sql = "insert into tb_article (typename,author,artquote) values(' 日 记 ',".$name.",".$saveid.")"; 
$num = $conne->uidRst($sql); 
if($num == 1X{ 

S$reback = "1"; 

}else{ 

$reback = '3'; 

} 

} 

}else{ 

S$reback = '2'; 


下 

}else if($act == 'delone'){ 

$delonesql = "delete from tb_review where id= ".$_GET[Trid]; 
$num = $conne->uidRst($delonesql); 

S$reback = "1'; 

}else if($act == 'delall 

$delallsql = "delete from tb_review where artid = ".$_GET['aid"]; 
$num = $conne->uidRst($delallsql); 

$reback = "1"; 


本 
echo $reback; 
3 


10.5 文章 管理 


10.5.1 文章 管理 概述 


对 一 个 博客 系统 来 说 ， 文 章 管理 是 最 基本 的 功能 ， 但 也 是 最 复杂 的 一 个 功能 。 文 章 管理 模块 包括 
发 表 文章 、 查 看 文章 、 删 除 文章 、 文 章 类 别 等 功能 ， 文 章 管理 中 发 表 文章 的 运行 结果 如 图 10.17 所 示 。 
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10.17 发 表 文 章 页 面 的 运行 结果 


10.5.2 发表 文章 功能 实现 过 程 


在 发 表 文章 页 addartphp 中 ， 创 建 一 个 发 表 文章 的 表单 ， 包 括 文章 标题 、 文 章 类 别 、 文 字 编辑 、 文 
章 内 容 等 表单 元 素 。 该 页 面 的 运行 结果 如 图 10.17 所 示 。 页 面 中 的 部 分 表单 元 素 如 表 10.1 所 示 。 


表 10.1 发 表 文 章 页 面 的 主要 元 素 属性 及 说 明 


元 素 名 称 元 素 属 性 说 了 明 
text type="text" id="txt_title" size="68" 文章 标题 
<?php 
foreach($typearr as $value){ 
?> 
Select <option value="<2php echo $value; ?>"><?php echo $value; ?></option> 文章 类 别 
<2php 
} 
?> 
name= "font" class="wenbenkuang" id="font" 
Select 文章 字体 
onChange="showfont(this.options[this.selectedIndex].value)" 
select class="wenbenkuang" onChange="showsize(this.options[this.selectedIndex].value)" 字体 大 小 
onChange="showcolor(this.options[this.selectedIndex].value)" name="color" size="1" 
select . | We 字体 辣 色 
class="wenbenkuang" id="select" 
textarea cols="75" rows="20" id="file" style="border:0px;width:520px;" 文章 内 容 
button id="enterbtn" onClick="retum chkart()" “提交 ”按钮 


当 用 户 填写 完 博客 主题 和 文章 内 容 后 ， 单 击 “提交 ”按钮 ， 将 触发 onclick 事件 调用 chkart() 函 数 。 
该 函数 将 文章 标题 、 文 章 类 别 等 相关 消息 进行 验证 后 ,传递 给 addart_chk.php 页 ， 根 据 该 页 的 返回 值 进 


_ 团 
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行 不 同 的 处 理 。chkart() 函 数 存储 于 jsvarticlejs 脚本 文件 中 ， 其 关键 代码 如 下 : 


function chkart(){ 

if($('txt_title').value==""}{ // 检 测 博 客 标题 
alert(" 博 客 标题 名 称 不 允许 为 空 ! "); 

$('txt_title').focus(); 

return false; 

$ 

if($ (file').value=="™"){ // 检 测 文 章 内 容 
alert(" 文 章 内 容 不 允许 为 空 !"); 

$('file').focus(); 

return false; 

} 

url = 'article/addart_chk.php?act=add8&title="+$("txt_title'). 
value+"&arttype="+$('articletype'). value+"& cont=" +$('file').value; /生成 URL 
xmlhttp.open('get',url,true); 

xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readystate == 4){ 

if(xmlhttp.status == 200){ 


msg = xmlhttp.responseText; 1/ 获取 返回 值 

if(msg == "1'){ 1/ 如 果 等 于 1， 发 表 成 功 
alert(' 文 章 发 表 成 功 '); 

xmilhttp.open('get','article/article.php',true); /刷新 文章 列表 


xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readystate == 4 && xmlhttp.status == 200){ 
$('showmenu').innerHTML = xmlhttp.responseText; 


办 
xmlhttp.send(null); 
}else if(msg == '2){ 
alert(' 添 加 失败 '); 


}else{ 
alert(msg); 


Wy 
xmlhttp.send(null); 


addart_chk.php 处 理 页 接 到 文章 信息 后 ， 将 新 信息 添加 到 数据 库 中 ， 如 果 成 功 ， 则 返回 1。 该 页 的 
代码 如 下 : 


<?php 

session_start(); /开启 Session 
header('content-type:text/html;charset=gb2312"); /设置 编码 
include "../Conn/conn.php"; // 载 入 数据 库 类 
Stitle=$_GET[title]; /文章 标题 
$typename = $_GET[arttype]; /文章 类 别 
$author=$_SESSION[name'; /| 当前 用 户 
$content=$_GET[ cont']; /文章 内 容 
$reback = '0'; 


$sql="Insert Into tb_article (typename ,title,content,author) 


@ 
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Values (".$typename.",".$title.",".$content.",”". $author.™")"; Winsert 语句 
$num = $conne->uidRst($sql); /添加 文章 
if($num == 1X{ // 如 果 添加 成 功 
$selectsql = "select upfile from tb_member where name = ".$author."" 

$artnum = $conne->getFields($selectsql,0); /得 到 上 传 文章 数量 
$upsql = "update tb_member set upfile = ".(++$artnum)." where name = ".$author.""; 

$num = $conne->uidRst($upsql); /更 新 

S$reback = "1"; 

}else{ 

S$reback = '2"; 

} 

echo $reback; 

?> 


10.5.3 ”文章 列表 功能 实现 过 


文章 列表 页 列 出 了 文章 的 类 别 、 标 题 和 点 击 率 。 该 页 实现 的 功能 包括 全 选 、 反 选 、 删 除 和 无 刷新 
分 页 。 次 则 国运 生 条 者 果 如 图 10.18 所 示 。 


图 10.18 ”文章 列表 页 面 的 运行 结果 
文章 列表 页 article.php 的 代码 如 下 : 


<?php 

session_start(); /开启 Session 
header('content-type:text/html;charset=gb23127); // 设 置 编码 
include_once '../conn/conn.php'; 

ifisset($_GET[act])) /判断 act 参数 
$act= $_GET[act]; 

Jelse{ 


$act="; 


i = $_SESSION[name'; /| 获取 当前 用 户 
$num = 10; // 每 页 显示 记录 数 
if(isset($_GET['curpage'])X{ /| 判断 是 否 为 当前 页 
$curpage = $_GET['curpage'; 

Jelse{ 
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$curpage =""; 
} 
if($act == 'del)f 


$rd=$_GET[rd]; 


$delsql = "delete from tb_article where id = -1 "; 


if(lempty($rd))\{ 

$delarr = explode(,,$rd); 
foreach($delarr as $value){ 

$delsql .= "or id = ".$value." "; 
$delnum = $conne->uidRst($delsql); 


} 


} 

if(empty($curpage) or is_null($curpage)){ 
S$curpage = 1; 

} 

if(empty($sql) or is_null($sql)){ 

// 创 建新 的 查询 语句 


// 如 果 是 删除 操作 
/得 到 要 删除 的 id 序列 
/初始 化 删除 语句 

/如 果 删 除 序列 不 为 空 
/| 拆 分 序列 转 为 数组 

/| 人 循环 数组 

// 更 新 删除 语句 


/执行 删除 语句 

// 如 果 是 第 一 次 打开 该 页 
/将 当前 页 变量 设 为 1 
/如 果 SQL 查询 语句 为 空 


$sql = "select id,typename ,title,hitnum,renum from tb_article where author = ".$name.” order by id desc"'; 


} 
$totnum = $conne->getRowsNum($sql); 
$totpage = ceil($totnum / $num); 


$tmpsql = $sql." limit ".($num *($curpage-1)).",".$num; 


$arr = $conne->getRowsArray($tmpsql); 
?> 


/| 总 记录 数 

/总 页 数 

/分 页 

/得 到 记录 数组 


上 述 代 码 实现 了 删除 数据 及 分 页 使 用 的 功能 ， 接 下 来 就 是 文章 列表 的 显示 。 代 码 如 下 : 


<div id = 'showarticle'> 


<table width="600" border="1" cellpadding="0" cellspacing="0" align="center"> 


<tr> 


<td height="25" width="30" align="center" valign="middle">&nbsp;</td> 


<td height="25" width="100" align="center" valign=' 


iddle"> 类 别 </td> 


<td height="25" width="370" align="center" valign="middle"> 标 题 </td> 
<td height="25" width="100" align="center" valign="middle"> 留 言 /点 击 率 </td> 


</tr> 

<?php 

foreach($arr as $key => $valueX{ 
?> 

<tr> 


<td align="center" valign="middle" height="25"> 


<input id="chk[<?php echo $key; ?>]" name="chk[]" type="checkbox" value="<?php echo $value['id]; ?>" /> 


</td> 


<td align="center" valign="middle" height=25>&nbsp;<?php echo $value['typename']; ?></td> 
<td align="left" valign="middle" height=25>&nbsp;<?php echo $value['title]; ?></td> 
<td align="center" valign="middle" height=25>&nbsp;<?php echo $value[renum]./.$value[hitnum]; ?> </td></tr> 


<?php 
让 


Ea 
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以 下 是 全 选 、 反 选 、 删 除 及 分 页 的 实现 。 删 除 则 是 全 选 、 反 选 操作 的 延续 。 实 现代 码 如 下 : 
if($totnum > OX 1/ 如果 有 记录 


3 
<tr> 
<td colspan="4" height="30" align="right" valign="middle"> 
<a href="#" onclick="retum alldel('<?php echo count($arr); ?>'")"> 全 选 </a> 
<a href="#" onclick="retum overdel(<?php echo count($arr) ?>")"> 反 选 </a> 
<! 一 删除 操作 ， 调 用 del() 函 数 ， 并 传递 了 两 个 参数 --> 
<button id="delbtn" name="delbtn" class="btn" ondlick="return del(‘<?php echo $_SERVER[SCRIPT_ NAME'; ?>", 
<?php echo count($arr); ?>',<?php echo $curpage; ?>)"> 删 除 选择 </button> 
<!-- 翻 页 超 链 接 ， 将 调用 pagination() 函 数 -> 
<?php echo ($curpage == 1)?" 首 页 ":" 
<a onclick='retum pagination(\".$_SERVER['SCRIPT_ NAMET." \",1)'> 首 页 </a>"; ?> 
<?php echo ($curpage==1)?" 上 一 页 ":" 
<a onclick='return pagination(\".$_SERVER ['SCRIPT_NAME'. \",".($curpage-1).")> 上 一 页 </a>"; ?> 
<?php echo ($curpage == $totpage)?' 下 一 页 ':” 
<a onclick='return pagination(\".$_SERVER ['SCRIPT_ NAME'"."\",".($curpage+1).")'> 下 一 页 </a>"; ?> 
<?php echo ($curpage==$totpage)?" 尾 页 ":" 
<a onclick='return pagination(\".$_ SERVER['SCRIPT_ NAME']."\",".($totpage).")> 尾 页 </a>"; ?> 
当前 是 第 <?php echo $curpage; ?> 页 / 共 <?php echo $totpage; ?> 页 <?php echo $totnum; ?> 条 记录 
<! 一 跳 转 操作 ， 调 用 jumpmem() 函 数 一 > 
跳 转 到 : <select id="jump" name="jump"”onchange="retum jumpmem(<?php echo $_SERVER[SCRIPT_ 
NAME'; ?>)"> 
<?php 
for($i=1,$i<=$totpage;$i++X{ 
if($i == $curpage) 
echo "<option value=".$i." selected>".$i."</option>"; 
else 
echo "<option value=".$i.">".$i."</option>"; 
} 
?> 
</select> 页 </td> 
</tr> 
<?php 


?> 


</table> 
</div> 


删除 、 翻 页 和 跳 转 操作 分 别 调用 了 脚本 文件 choosejs 中 的 3 个 函数 。 删 除 函 数 del0 有 3 个 参数 ， 
这 3 个 参数 分 别 代表 删除 处 理 页 的 URL、 当 前 页 的 记录 总 数 和 当前 页 的 页 码 。 

函数 中 首先 声明 了 一 个 数组 rd， 然 后 循环 判断 当前 页 中 的 复 选 框 状 态 ， 将 要 删除 的 复 选 框 的 vlaue 
值 保存 到 rd 中 ; 接着 将 rd 和 当前 页 的 页 码 传 给 处 理 页 ; 最 后 调用 rerst() 函 数 将 处 理 页 的 输出 结果 显示 
到 指定 区 域 。choose.js 文件 的 删除 代码 如 下 : 


/删除 所 选 
function del(chkurl,tot,curpageX{ 
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var rd=new Array(); /声明 数组 
for(i=0,j=0;i<tot;it+X{ // 循 环 判断 复 选 框 
if(l$Cchk[+it]).checked)f 

Jelse{ // 如 果 复 选 框 被 选中 

rd] = $(Cchk[+i+]).value; // 将 value 值 保存 到 rd 数组 中 
ts 

} 

} 

if(rd == "}{ // 如 果 没有 删除 ， 提 示 错 误 
alert(' 请 选取 要 删除 的 数据 !"); 

return false; 

} 

url = chkurl+"?act=del&rd="+rd+'&curpage='+curpage; /| 生成 URL 
xmlhttp.open('get',url,true); // 创 建新 请 求 
xmlhttp.onreadystatechange = rerst; // 调 用 rerst() 函 数 
xmlhttp.send(null); 

return false; 

} 


function rerst(X{ 
if(xmlhttp.readystate == 4){ 


if(xmlhttp.status == 200){ /判断 请 求 是 否 完成 
$(showarticle'),innerHTML = xmlhttp.responseText'; /将 返回 值 显示 到 指定 区 域 
} 

} 

} 


对 翻 页 的 处 理 统一 使 用 pagination() 函 数 ， 该 函数 有 两 个 参数 : 一 个 是 处 理 页 的 URL; 另 一 个 是 当 
前 页 的 页 码 。 代 码 如 下 : 


// 分 页 

function pagination(chkurl,curpage){ 

url = chkurl+'?act=next&curpage='+curpage; /生成 URL 
xmlhttp.open(get,urltrue); // 创 建新 请 求 
xmlhttp.onreadystatechange = rerst; // 调 用 rerst() 函 数 
xmlhttp.send(null); 

return false; 


跳 转 是 翻 页 的 另 一 种 形式 。 翻 页 只 能 一 页 一 页 地 翻 ， 而 跳 转 可 以 直接 跳 到 指定 的 页 面 。 代 码 如 下 : 


// 跳 转 

function jumpmem(chkurl){ 

jumppage = $(jump').value; // 获 取 跳 转 页 数 
url = chkurl+'?curpage='+jumppage; /生成 URL 
xmlhttp.open('get,urltrue); // 创 建 请 求 
xmlhttp.onreadystatechange = rerst; // 调 用 rerst() 函 数 
xmlhttp.send(null); 

return false; 
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10.5.4 文章 类 别 功能 实现 过 程 
文章 类 别 模块 实现 文章 类 别 的 添加 、 删 除 操作 。 文 章 类 别 页 面 的 运行 结果 如 图 10.19 所 示 。 


10.19 文章 类 别 页 面 的 运行 结果 
文章 的 类 别 被 保存 到 用 户 信息 中 ， 默 认 有 一 个 “上 日记” 类 别 ， 当 有 多 个 类 别 时， 将 使 用 去 号 分 割 。 
显示 文章 类 别 时 ， 首 先 获 取 该 用 户 的 类 别 字段 的 值 ， 将 字符 串 使 用 逗号 分 割 成 数组 ， 最 后 由 foreach 循 
环 显示 所 有 类 别 。 文 章 类 别处 理 文件 arttype.php 的 代码 如 下 : 


<?php 

session_start(); /开启 Session 
header('content-type:text/html;charset=gb2312'); /| 设置 编码 

include_once '../conn/conn.php'; // 载 入 数据 库 类 文件 

$act = $_GET[act]; /获取 act 值 

$name = $_SESSION[name'; /获取 用 户 名 称 

$sql = "select arttype from tb_member where name = '".$name.""; /生成 查询 语句 
$typefields = $conne->getFields($sql,0); // 获 取 字 段 值 

32 

<table id="addtypefm" width="300" border="1" cellspacing="0" cellpadding="0" align="center"> 
<tr> 

<td colspan="2" height="25" align="center" valign="middle"> 文 章 类 别管 理 </td> 

</tr> 

<?php 

$typearr = explode(',', $typefields); // 使 用 加 号 分 割 字符 串 存 为 数组 
foreach($typearr as $valueX{ /循环 输出 数组 

人 

<tr> 


<td width="200" height="25" align="center" valign="middle">&nbsp;<?php echo $value; ?></td> 

<td height="25" align="center" valign="middle">&nbsp; 

<button id="deltype" value="<?php echo $value; ?>" class="btn" 

onclick="return delarttype('<?php echo $typefields; ?>'",'<?php echo $value;?>")"> 删 除 </button></td> 
</tr> 

<?php 

上 


?> 
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<tr> 

<td width="200" height="25" align="center valign="middle"> 

<input id="addarttype" name="addarttype" type="text" class="txt" /></td> 

<td height="25" align="center’ valign="middle"><button id="addtype"” value="<?php echo $value; ?>" 
class="btn" 

onclick="return addarttype('<?php echo $typefields; ?>')"> 添 加 新 类 </button></td> 

</tr> 

</table> 


添加 文章 类 别 的 操作 是 用 户 单 击 “ 添 加 新 类 ”按钮 后 ， 将 调用 addarttype() 函 数 ， 并 将 原 类 别 字段 


传 给 该 函数 。addarttypeO 函 数 将 创建 一 个 新 请 求 ， 并 根据 请 求 的 输出 值 做 不 同 的 处 理 。addarttypeO) 函 数 
存储 于 jsvarticlejs 脚本 文件 中 ， 其 代码 如 下 : 


/添加 文章 类 别 

function addarttype(typestr){ 

if($(‘addarttype').value == "X{ // 判 断 类 别名 称 是 否 为 空 
alert(' 请 输入 添加 类 别名 称 '); 

S$('addarttype').focus(); 

return false; 


} 

url = 'article/arttype_chk.php?act=add&typestr='+typestr+'&typename='+$(addarttype).value ; 
xmilhttp.open('get ,url,true); // 创 建新 请 求 
xmlhttp.onreadystatechange = function(){ 

if(xmlhttp.readystate == 4){ 

if(xmlhttp.status == 200){ 

msg = xmlhttp.responseText; // 获 取 处 理 页 输出 结果 
// 根 据 返 回 的 不 同 结 果 ， 做 不 同 的 处 理 

if(msg == '2{ 

alert(' 添加 失败 ); 

}else if(msg == "3'){ 

alert(' 添 加 名 称 重复 '); 

}else if(msg == '1){ 

showarttype(); 

jelsef 

alert(msg); 


} 
} 


} 
xmlhttp.send(null); 


arttype_chk.php 是 addarttypeO 函 数 调用 的 处 理 页 ， 将 获取 到 的 原 类 别 字段 后 面 加 上 逗号 和 新 类 别 
名 称 ， 再 将 该 字段 保 在。 代码 如 下 : 


<?php 

session_start(); /开启 Session 支持 
header('Content-Type:text/html;charset=gb2312'); /| 设置 编码 
include_once '../../conn/conn.php'; // 载 入 数据 库 类 


SS 
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$act = $_GET[act]; /获取 act 值 

$typestr = $_GET[typestr]; // 原 类 别 字符 串 
$typename = $_GET[typename]; // 要 获取 的 类 别 字符 串 
$arr = explode(',', $typestr); // 将 原 字 符 串 转 为 数组 
S$reback = '0'; 

$name = $_SESSION[name'; // 获 得 当前 用 户 

if($act == 'add'){ // 如 果 是 添加 操作 
iflin_array($typename,$arr)){ /| 判断 名 称 是 否 重复 
$reback = '3'; 

Jelse{ 

$typestr .= ",".$typename; // 将 新 类 别名 称 添加 到 原 类 别 后 面 
$sql = "update tb_member set arttype = ". $typestr." where name = ".$name.""; 
$num = $conne->uidRst($sql); /执行 更 新 语句 
if($num == 1X{ 

$reback = "1'; 

}else{ 

$reback = '2'; 

二 

} 

} 


删除 文章 类 别 和 添加 类 别 大 同 小 异 ， 但 要 注意 两 点 第 一 不 允许 删除 默认 的 “日 记 ” 类别， 如 果 
删除 的 是 “日 记 ” 类 别 ， 则 弹出 警告 ， 第 二 将 删除 的 类 别 下 的 所 有 文章 ， 全 部 放 到 默认 的 “日 记 ” 目 
录 下 。 删 除 文章 类 别 操作 由 arttype_chk.php 文件 完成 ， 其 代码 如 下 : 


// 添 加 类 别 代码 
else if($act == 'del'X{ 
if($typename == ' 日 记 'X{ 


$reback = '3"; 
}else{ 
S$tmpstr = 日 记 '; 


for($i=1;$i<count($arr);$i++){ 
if($arr[$i] != $typename}{ 
$tmpstr .= ",".$arr[$i]; 

} 


$changsql = "update tb_article set typename = ' 日 记 'where author = ".$name." and typename=". $typename.""; 
$num = $conne->uidRst($changsql); 

$sql = "update tb_member set arttype = ".$tmpstr." where name = ".$name."™; 

$num = $conne->uidRst($sql); 


if($num == 1X{ 
$reback = "1"; 
}else{ 
S$reback = '2"; 
} 

echo $reback; 
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10.6 用 户 管 理 


10.6.1 用户 管理 概述 


用 户 管理 模块 是 后 台 的 一 个 主要 模块 ， 它 是 用 来 管理 注册 用 户 的 ， 有 删除 用 户 、 冻 结 用 户 、 查 看 
博客 点 击 率 、 查 看 所 有 用 户 信息 等 多 项 功能 。 用 户 管理 模块 的 运行 结果 如 图 10.20 所 示 。 
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图 10.20 用 户 管理 模块 的 运行 结果 


用 户 管理 模块 是 博客 后 台 系 统 所 包含 的 6 个 模块 之 一 。 其 他 的 模块 依次 为 文章 管理 模块 、 图 片 管 
理 模块 、 管 理 员 设置 模块 、 系 统 日 志 模块 以 及 数据 备份 模块 。 

用 户 管理 模块 中 包含 的 功能 有 查询 博客 、 推 荐 博客 、 冻 结 博客 、 查 看 详细 资料 和 删除 博客 等 功能 。 
其 中 推荐 博客 和 冻结 博客 的 实现 方法 基本 相同 ， 而 查看 详细 资料 和 删除 博客 在 前 面 也 接触 过 ， 这 里 只 
讲解 查询 博客 和 推荐 博客 的 实现 方法 。 


10.6.2 ”用 户 管理 功能 实现 过 程 


首先 来 看 一 下 用 户 管理 模块 的 首页 member.php。 在 member.php 文件 中 ， 将 数据 库 中 存储 的 用 户 
信息 进行 分 页 输出 ， 并 且 完 成 各 种 附加 功能 的 设置 。 其 运行 结果 如 图 10.21 所 示 。 


编号 用 户 号 mail 真实 姓名 性别。 博客 点 击 言 推荐 车 结 / 希 东 详细 资料 
所 eonsk 更 0 否 浆 闭 详细 资料 
回 2 mrlx 男 否 攻 结 详细 资料 
23 zlll 男 0 到 冻结 详细 资料 
四。 zz 4 格格 。 Tesszaea em 王仁 而 女 3 井 4 详细 资料 
回 21 博客 皇帝 111141199ee com 。 张 其 卫 田 5 推荐 冻结 详细 资料 
20 ny DG 。 732565048q9e em 小 责 男 5 否 东周 详细 资料 
回 19 ILOVE YOU 845129549aq8 CoN 123 安 5 到 革 结 详细 资料 

2 FE 过“ 硬 到 EE 珊 首页 上 一 页 下 一 页 尾 页 当前 是 第 ! 页 / 共 !1 页 7 条 记录 跨 转 到 : 1 ”页 


图 10.21 用 户 管理 页 面 的 运行 结果 
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member.php 页 的 代码 如 下 : 

<?php 

session_start(); /开启 Session 
include_once '.Jconm/conn.php'; // 载 入 数据 库 类 

?5 

<script language="javascript" src="../js/choose.js"></script> ”// 载 入 js 文件 
<?php 

$num = 10; /每 页 显示 记录 数 
if(empty($_GET['act])X{ // 获 取 当 前 是 第 几 页 
$act="™"; 

}else{ 

$act = $_GET[act]; // 当 前 页 

} 

$sql="; //SQL 语句 
if(empty($curpage) or is_null($curpage)){ // 获 取 当 前 是 第 几 页 
$curpage = 1; 

}else{ 

$curpage = $_GET[curpage]; // 当 前 页 

} 

if(empty($fields) or is_null($fields)X{ /排列 顺序 ， 默 认为 id 
S$fields = "id'; 

Jelse{ 

S$fields = $_GET[Yields']; /依照 该 字段 排列 记录 


} 

// 是 否 有 查询 内 容 ， 如 果 没 有 

if(!isset($_REQUEST[cont])X 

// 直 接生 成 SQL 查询 语句 ， 使 用 $fields 字段 作为 排序 依据 

$sql = "select id,name,email,realname,sex,birthday,hitnum,freeze,isnominate from tb_member order by 
".$fields." desc"; 

// 如 果 有 查询 内 容 ， 就 获取 要 查询 的 字段 、 查 询 方式 和 查询 内 容 
}else{ 

$querymem = $_REQUEST[querymem']; 

$signslt = $_REQUEST[signslt]; 

$cont = $_REQUEST[cont]; 

// 初 始 化 变量 $sql 为 一 个 无 条 件 的 查询 语句 

$sql = 'select id,name,emailrealname,sex,birthday,hitnum,freeze,isnominate from tb_member '; 
// 根 据 查询 字段 所 在 的 组 ， 生 成 不 同 的 where 子 名 
if(in_array($querymem,array('id",'upfile','uppics','hitnum','birthday'))X{ 
$sql .= " where ".$querymem." ".$signslt." ".$cont; 

}else if(in_array($querymem,array('name','realname','qq','email'))X{ 
if($signslt == 'exac'}{ 

$sql .= "where ".$querymem." = ".$cont.""; 

}else if($signslt == 'mist'X{ 

$sql .= "where ".$querymem." like '%".$cont."%™; 

} 


} 
/添加 order by 子 句 
$sql .= " order by ".$fields." desc"; 
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} 
S$totnum = $conne->getRowsNum($sql); /| 记录 数 
S$totpage = ceil($totnum / $num); /页 数 


$tmpsql = $sql." limit ".($num *($curpage-1)).",".$num; 
$arr = $conne->getRowsArray($tmpsql); 
?> 


10.6.3 ”查询 博客 功能 实现 过 程 


查询 博客 是 根据 博客 不 同 的 信息 来 查询 特定 的 博客 。 例 如 id 号 查询 、 点 击 率 查询 等 。 查 询 页 是 单 
独 的 一 页 ， 被 包含 在 member.php 页 中 。 当 用 户 查 询 时 ， 该 页 将 查询 字段 、 查 询 方式 和 查询 内 容 传 给 
member.php 页 ， 再 由 member.php 页 进行 处 理 ， 该 页 的 处 理 过 程 在 上 面 已 经 介绍 过 。 查 询 页 query.php 
的 代码 如 下 : 


<?php 

include_once '../conn/conn.php’; /| 载 入 数据 库 类 
?> 

<script language="javascript" src="../js/query.js"></script> 

<table border="0" cellpadding="0" cellspacing="0" align="center"> 

<form id="queryfm” name="queryfm" method="post” action="member.php?act=query” onSubmit="return 
chkquery()"> 

<tr> 

<td width="50" height="25"> 查 询 : </td> 

<td width="100"> 

<select id="querymem" name="querymem'" onChange="return changequery()"> 
<option value="id"> 编 号 </option> 

<option value="name"> 账 号 </option> 

<option value="realname"> 真 实 姓 名 </option> 

<option value="email">email</option> 

<option value="qq">qq</option> 

<option value="birthday"> 出 生日 期 </option> 

<option value="upfile"> 文 章 数 </option> 

<option value="uppics"> 上 传 图 片 </option> 

<option value="hitnum"> 点 击 率 </option> 

</select> 

</td> 

<td width="50" align="center" valign="middle"><div id="sign"> 

<select id="signslt" name="signslt"> 

<option value="&gt;"> 大 于 </option> 

<option value="="> 等 于 </option> 

<option value="&lt;"> 小 于 </option> 

</select> 

</div></td> 

<td width="100"><div id="input"><input id="cont" name="cont" type="text" size="30" /></div></td> 
<td><input id="querybt" name="querybt" type="submit" value=" 查 询 "></td> 
</tr> 

</form> 

</table> 


全 


时 ， 
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在 查询 页 中 ， 当 用 户 选择 不 同 的 查询 条 件 时 ， 查 询 方式 也 会 跟着 发 生变 化 。 例 如 查询 id、 点 击 率 
查询 条 件 有 大 于 、 等 于 和 小 于 ， 当 查询 账号 、 真 实 姓名 、E-mail 时 ， 查 询 条 件 变 为 精确 和 模糊 。 


这 是 通过 query.js 脚本 来 控制 的 ， 实 现代 码 如 下 : 


10. 


function changequery(){ 

key = document.getElementByld(‘querymem').value; /获取 查询 条 件 
Switch(key){ 

让 当 查 询 条 件 和 数字 有 关系 时 */ 

case id': 

case 'upfile': 

case 'uppics': 

case hitnum : 

document.getElementByld('sign').innerHTML = '<select id="signslt" name="signslt"> 
<Option value= "&gt"> 大 于 </Option><option value="="> 等 于 </option><option value="&lt"> 小 于 </option></select>'; 
document.getElementByld('cont').value="; 

break; 

让 当 查 询 条 件 和 字符 串 有 关系 时 */ 

case name' 

case realname': 

case 'email': 

case 'qq': 

case ‘title': 

case 'author : 

case 'typename': 

case "upname': 

case "upauthor : 

case 'operator : 

document.getElementByld('sign').innerHTML = '<select id="signslt" name="signslt"> 
<Option value= "exac"> 精 确 </Option><option value="mist"> 模 糊 </option></select>'; 
document.getElementBylId('cont').value="; 

break; 

人 * 当 查询 条 件 和 时 间 有 关系 时 */ 

case 'birthday": 

case firsttime': 

case uptime': 

document.getElementByld('sign').innerHTML = '<select id="signslt" name="signslt"> 
<Option value= "&gt"> 大 于 </Option><option value="="> 等 于 </option><option value="&lt"> 小 于 </option></select>'; 
document.getElementByld('cont').value=' 查 询 格式 : YYYY-MM-DD'; 

break; 


/* 省 略 部 分 为 其 他 模块 的 查询 方式 设置 */ 


} 
document.getElementByld('cont').focus(); 


document.getElementByld('cont').select(); 


6.4 推荐 博客 功能 实现 过 程 
当 用 户 为 推荐 博客 时 ， 该 用 户 的 博客 空间 将 在 首页 的 显著 位 置 显示 ， 这 样 可 以 提高 网 站 和 个 人 空 


27n 


间 的 
型 为 
文件 
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人 气 和 访问 量 。 推 荐 博客 的 实现 方法 非常 简单 ， 在 用 户 表 中 设置 一 个 字段 ,名 称 是 isnominate， 类 
int。 当 用 户 注册 时 , 字段 的 默认 值 为 0, 当 设 为 推荐 博客 时 , 将 该 字段 设 为 1。 该 操作 从 member.php 
中 设置 的 超 链接 开始 ， 其 代码 如 下 : 


<a href="member_chk.php?act=nominate&id=<?php echo $value['id]; ?>&isnominate=<?php echo S$value 
Tisnominate]; ?>"><?php echo ($value[isnominate']==0?" 否 "…" 推 荐 "); ?></a> 


链接 到 member_chk.php 文件 , 根据 博客 用 户 的 id 号 和 isnominate 字段 的 值 来 更 新 用 户 表 , 代码 如 下 : 


10. 


<?php 

session_start(); 
header('Content-Type:text/html;charset=gb2312'); 
include_once '../conn/conn.php'; 

$act = $_GET[act]; 


if($act == 'del'X{ 

思 // 删 除 操作 

}else if($act == "nominate')f /推荐 博客 

$id=$_GET['id]; // 获 取 用 户 id 号 

$nominate = ($_GET[isnominate]+1) % 2; // 将 获取 的 字段 值 加 1 再 取 模 
$sql = "update tb_member set isnominate = ".$nominate." where id = ".$id; 

$cont = 推荐 用 户 操作 ， 账 号 :'.$id; /系统 日 志 内 容 

$num = $conne->uidRst($sql); // 更 新 字段 值 
if(lempty($num)X 

S$inssql = "insert into tb_log values(",".$cont.",".$_ SESSION[manager].",now())”; 
$num = $conne->uidRst($inssql); /添加 日 志 


echo "<script>location='member.php';</script>"; 


} 


10.7 技术 提炼 


7.1 文件 包含 技术 


通过 文件 包含 技术 实现 文件 的 包含 ， 有 助 于 网 站 页 面 的 规划 、 管 理 和 更 新 。 例 如 ， 网 站 的 头 文件 、 


尾 文件 和 数据 库 的 连接 文件 ， 都 可 以 通过 包含 的 方式 进行 载 入 ， 这 样 就 可 以 避免 在 每 个 页 面 中 编写 相 


同 的 


代码 ， 即 减少 代码 的 元 余 ， 同 时 也 便于 对 这 些 内 容 的 修改 和 更 新 。 
在 博客 模块 的 个 人 首页 中 通过 include_once() 函 数 包含 文件 ,使 用 include_once() 函 数 在 包含 文件 时 ， 


如 果 该 文件 中 的 代码 已 经 被 包含 ， 则 不 会 被 再 次 包含 。include_once0 函 数 应 用 于 脚本 执行 期 间 同 一 个 
文件 有 可 能 被 包含 超过 一 次 的 情况 下 ， 以 确保 它 只 被 包含 一 次 ， 从 而 避免 出 现 函数 重 定义 、 变 量 重新 
赋值 等 问题 。 


全 
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博客 个 人 首页 的 左 侧 包 含 文件 代码 如 下 : 


<!- 左 侧 --> 

<div id="leftfloatcont"> 

<div id="personinfo"> 

<?php include_once 'personinfo.php'; ?> 
</div> 

<div id="date"> 

<?php include_once 'date.html'; ?> 
</div> 

<div id="showmessdiv"> 

<?php include_once 'mess1.php'; ?> 
</div> 

</div> 


10.7.2 ”SQL 查询 技术 

在 我 的 文章 模块 中 ， 总 体 是 采用 SQL 语句 查询 技术 ， 从 数据 库 中 根据 查询 条 件 完成 数据 的 循环 输 
出 。 其 应 用 到 的 SQL 技术 如 下 : 

1. 排序 功能 


所 谓 的 排序 功能 就 是 先 来 后 到 之 说 ， 排 序 分 为 升序 和 降序 ， 升 序 是 从 小 到 大 ， 而 降序 则 相反 ， 是 
从 大 到 小 。 排 序 主要 有 两 个 关键 字 : 升序 asc 和 降序 desc。 其 语法 格式 如 下 : 


select * from 表 名 order by 字段 名 asc/desc 

2. 查 记 录 总 数 

查 记录 总 数 就 是 将 数据 库 中 的 记录 总 数 查 询 出 来 。 查 询 记录 使 用 的 关键 字 是 count 和 as。 其 语法 
如 下 : 

select count (*) as ( 自 定义 字段 ) from 表 名 

3. 条 件 查询 


所 谓 的 条 件 查询 就 是 指定 其 中 的 一 条 或 多 条 查询 ， 它 是 带 条 件 的 ， 这 里 所 用 的 关键 字 是 where。 其 
语法 格式 如 下 : 


select * from 表 名 where 字段 = 条 件 


下 面 讲解 一 下 这 些 SQL 查询 技术 在 本 模块 中 的 具体 应 用 。 
(1) 我 的 文章 中 应 用 的 查询 语句 的 代码 如 下 : 


$sql = "select * from tb_article where author = ".$_GET[uid]." order by id desc "; 
S$totnum = $conne->getRowsNum($sql); 

S$totpage = ceil($totnum / $num); 

S$tmpsql = $sql." limit ".($num *($curpage-1)).",".$num; 
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$arr = $conne->getRowsArray($tmpsql); 

$conne->close_rst(); 

查询 文章 表 (tb_article) 中 的 所 有 记录 ， 查 询 条 件 是 author 的 值 与 超 链接 传递 的 值 相同 ， 并 以 id 
为 条 件 进行 降序 排列 。 最 后 输出 查询 结果 ， 完 成 我 的 文章 标题 、 发 表 时 间 、 部 分 文章 内 容 的 循环 输出 。 

(2) 文章 浏览 中 应 用 的 查询 语句 的 代码 如 下 : 


$showartsql = "select * from tb_article where id = ".$_GET[artid']; /| 生成 查询 语句 
$artarr = $conne->getRowsArray($showartsql); // 返 回 数组 形式 的 记录 
$conne->close_rst(); // 清 空 结果 集 


以 超 链接 中 传递 的 文章 的 id 为 条 件 ， 执 行 查询 操作 ， 查 询 的 结果 就 是 有 关 这 条 记录 的 详细 信息 ， 
即 文章 的 详细 内 容 。 


10.7.3 会 话 处 理 技术 


想 要 使 用 文章 管理 模块 ， 前 提 是 用 户 必 须 登 录 ， 匿 名 用 户 是 无 法 访问 这 些 功 能 的 ; 想 要 删除 文章 
和 评论 ， 前 提 是 当前 用 户 要 么 是 管理 员 权限 ， 要 么 是 文章 拥有 者 ， 和 否则 不 会 显示 删除 功能 。 这 两 方面 
的 控制 都 需要 Session 的 配合 ， 本 节 就 来 讲 一 下 Session 的 应 用 及 常见 的 问题 处 理 。 

Session 的 中 文 译名 为 “会 话 ”， 是 指 用 户 从 进入 网 站 开始 ， 直 到 关闭 网 站 这 段 时 间 内 ， 所 有 网 页 
共同 使 用 的 公共 变量 的 存储 机 制 。Session 比 Cookie 更 有 优势 : Session 是 存储 在 服务 器 端的 ， 不 易 被 
伪造 ，Session 的 存储 没有 长 度 限制 ，Session 的 控制 更 容易 等 。 

PHP 主要 是 通过 会 话 (Session〉 处 理 函数 来 对 Session 进行 控制 和 使 用 。 常 用 的 处 理 函 数 如 表 10.2 
所 示 。 

表 10.2 PHP 常用 的 会 话 处 理 函 数 


函数 函数 说 明 
session start(); 开启 Session 或 返回 已 经 存在 的 Session 
$ SESSION[mame'] = value: | 注册 一 个 Session 变量 
session_id() 设 定 或 取得 当前 的 session_id 值 


检测 指定 的 Session 值 是 否 存 在 。Isset() 函 数 不 只 可 以 检测 Session, 还 可 以 检测 其 他 类 


Pn eh 型 ， 如 isset($_POST['name']、isset($_GET['name']) 等 


session_regenerate_id|) 更 改 session_id 的 值 

Session_nal 返回 或 改变 当前 Session 的 name 
unset($_ SESSION[name']) 删除 名 为 name 的 Session 
Session_destroyO 结束 当前 会 话 ， 删 除 所 有 Session 


ee 
人 注意 (1) 如 果 要 改变 当前 Session 的 name 值 ， 必 须 在 session() 之 前 调用 session name() 函 数 ， 
而 且 session_ name 不 能 全 部 是 数字 ， 否 则 会 不 停 地 生成 新 的 session id。 
(2) 不 可 以 写成 unset($_SESSION)， 这 样 会 禁止 整个 会 话 的 功能 。 
@ 
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下 面 介绍 几 个 使 用 Session 时 要 注意 的 问题 。 

1. 尽 可 能 地 将 session_start() 放 到 第 1 行 

这 种 情况 是 新 手 最 容易 犯 的 错误 。 产 生 的 错误 代码 如 下 : 

Warning: session_start() function session-stam: Cannot send session cache limiter - headers already sent-. 

其 原因 就 是 在 使 用 session_start(0) 之 前 ， 有 HTML 代码 输出 。 也 许 有 的 读者 会 说 : session_start() 之 
前 没有 任何 代码 ， 绝 对 没有 ， 但 是 还 是 出 现 了 这 个 错误 。 那 么 ， 请 检查 你 的 程序 ， 是 不 是 有 空 行 ， 或 


类 似 echo 语句 的 输出 。 如 果 有 ， 请 去 掉 ， 因 为 就 算是 一 个 小 小 的 空格 都 是 不 可 以 的 。 所 以 为 了 避免 这 
类 错误 的 发 生 ， 尽 可 能 地 将 session_start0 放 到 第 1 行 。 


2. 在 使 用 Session 之 前 一 定 要 先 写 session_start() 


大 多 数 读者 在 使 用 Session 之 前 都 先 调用 session_start0 函 数 ， 但 对 于 session_destroy() 函 数 却 经 常 
忽略 。session_destroy() 函 数 虽然 是 结束 当前 会 话 并 删除 所 有 Session, 但 在 删除 之 前 , 也 要 先 开 启 Session 
支持 才 可 以 ， 和 否则 会 产生 如 下 的 错误 代码 : 

Session_destroy() [function.session-destroy]: Trying to destroy uninitialized session in… 


所 以 ， 凡 是 在 使 用 Session 函数 的 页 面 中 ， 都 要 加 上 session_start() 这 人 句 话 。 
3. 删除 所 有 Session 


如 果 想 删除 所 有 Session， 但 又 不 想 结束 当前 会 话 ， 用 unset 一 个 一 个 删除 实在 是 太 麻烦 了 ， 最 简 
单 的 办 法 就 是 将 一 个 空 数组 赋 给 $_SESSION， 如 $_SESSION = array()， 这 样 就 解决 了 。 


10.7.4 ”冻结 、 解 冻 技术 


所 谓 用 户 管理 冻结 技术 ， 就 是 将 用 户 冻 结 后 ， 用 户 将 无 法 登录 ， 自 然 有 冻 就 有 解 ， 如 果 解 冻 ， 用 
户 便 可 以 正常 登录 。 用 户 冻 结 技术 主要 是 针对 一 些 不 守 网 站 规则 的 用 户 而 建立 的 功能 。 其 实用 户 冻 结 
技术 与 用 户 推 荐 、 文 章 审核 等 技术 的 实现 原理 是 相同 的 。 

采用 数据 库 条 件 式 方法 ， 在 指定 数据 库 内 设置 单独 的 字段 作为 冻结 技术 的 条 件 ， 当 条 件 为 真 时 用 
户 正 常 运 行 解冻 ， 当 条 件 为 假 时 用 户 停止 运行 冻结 ， 这 就 是 冻结 和 解冻 的 原理 。 本 模块 中 具体 的 冻结 、 
解冻 操作 是 在 adminmembermember.php 文件 中 完成 的 ， 其 代码 如 下 : 


$id = $_GET[id]; 
S$freeze = ($_GET[fz]+1) % 2; 

$sql = "update tb_member set freeze = ".$freeze." where id = ".$id; 

$cont = ' 冻 结 (解冻 ) 操 作 ， 账 号 :'.Sid; 

$num = $conne->uidRst($sq)); 

if(lempty($num)X{ 

Sinssql = "insert into tb_log values(",".$cont.",".$_SESSION[manager].",now())"; 
$num = $conne->uidRst($inssql); 


辐 
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10.7.5 配置 文件 


在 配置 文件 中 设置 了 服务 器 的 目录 、 博 客 目录 、 上 传 图 片 目录 、 图 片 大 小 上 限 等 信息 。 配 置 文件 
config.php 存储 于 前 台 。 配 置 文件 的 代码 如 下 : 


<?php 

define(CPATH',$_SERVER[DOCUMENT_ROOT]); // 服 务 器 目录 
define(ROOT',/db_blog1.0 /"); // 情 客 目 录 
define(ADMIN','admin/"); /后 台 目 录 
define('PIC','center/pics/image/"); // 上 传 图 片 目录 
define('BAK','sqlbak/"); /| 备份 目录 
define('HEADGIF','headgif/"); /头像 目录 
define('MYSQLPATH','D:\xampp\MySQL\\binW); /MySQL 执行 文件 路 径 
define(MYSQLHOST','127.0.0.1"); /WMySQL 服务 器 IP 
define(MYSQLDATA','db_blog'); /MySQL 数据 库 
define('MYSQLUSER','root'); /MySQL 账号 
define(MYSQLPWD', 111); /MySQL 密码 
define('MAXSIZEPIC',500000); // 允 许 上 传 的 图 片 的 最 大 字 节 权 
?> 


10.7.6 ”公共 函数 


博客 系统 的 公共 函数 文件 包括 返回 指定 文件 夹 下 的 文件 列表 和 判断 上 传 文件 的 后 级 。 其 存储 于 
centerinc\func.php 中 ， 函 数 代码 如 下 : 


<?php 

1/ 返回 文件 夹 下 的 文件 列表 

function show _file($f_name)}{ 

$d_open = opendir($f_name); 1/ 打开 指定 目录 

$num = 0; 

while($file = readdir($d_open))f // 循 环 输出 目录 下 的 文件 
$filename[$num] = $file; // 将 文件 存 成 数组 


$num++; 


} 

closedir($d_open); /关闭 目录 
return $filename; // 返 回 文件 数组 
} 

六 

判断 文件 后 组 

$f_type: 允许 文件 的 后 缀 类 型 

$f_upfiles: 上 传 文件 名 

村 
function f_postfix($f_type,$f_upfiles){ 


282 
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$is_pass = false; 


$tmp_upfiles = split("\.",$f_upfiles); // 将 后 缀 字符 串 分 给 数组 
$tmp_num = count($tmp_upfiles); /获取 数组 大 小 
if(in_array(strtolower($tmp_upfiles[$tmp_num - 1 ),$f_type)) ”// 上 传 的 后 缀 是 否 在 允许 范围 
Sis_pass = $tmp_upfiles[$tmp_num - 1]; // 保 存 后 缀 

return $is_pass; // 返 回 结果 

} 

?> 


10.8 本 章 小 结 


本 章 的 博客 管理 系统 首先 介绍 了 博客 的 基本 概念 、 发 展 前 景 、 影 响 范围 及 博客 网 的 功能 分 类 ， 使 
读者 对 当今 主流 博客 有 了 一 个 大 致 的 认识 。 其 次 ， 实 现 了 一 个 博客 系统 ， 包 含 所 有 基本 功能 的 项 目 开 
发 ， 使 读者 对 如 何 开发 一 个 博客 网 站 有 了 一 个 初步 的 了 解 。 最 后 ， 希 望 读者 通过 自己 的 努力 ， 来 逐步 


完善 和 加 强 这 个 博客 网 的 实用 功能 ， 最 终 达 到 一 个 令 自 己 满意 的 作品 。 
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创建 PDF 文档 模块 


(PHP+MySQL 实现 ) 


PDF 文档 格式 是 当前 比较 流行 的 电子 文档 与 电子 表格 的 一 种 标准 
格式 。PDF 提供 了 完善 的 压缩 处 理 ， 无 论 创 建 者 创建 的 PDF 文档 使 用 
何 种 字体 .什么 样 的 图 片 与 版 式 设计 ,浏览 者 都 可 以 通过 免费 的 Adobe 
Reader 对 其 进行 阅读 。 本 章 开 发 一 个 支持 PDF 文档 创建 和 下 载 功 能 
的 文章 管理 模块 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

MW ”PHP 国际 化 

MW FPDF 的 安装 与 配置 

MW PDF 文档 的 创建 

MW 向 PDF 中 插入 图 片 

H 在 PDF 中 绘制 表格 

MW 在 PDF 中 输出 中 文 

MW PDF 文档 的 下 载 
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11.1 创建 PDF 文档 模块 概述 


11.1.1 ”模块 概述 


创建 PDF 文档 模块 的 重点 是 将 用 户 添加 的 文章 生成 PDF 文档 ， 并 且 完 成 PDF 文档 的 下 载 操 作 。 
其 中 还 实现 了 文章 添加 、 文 章 浏览 等 辅助 功能 ， 并 且 实 现 网 站 的 国际 化 。 既 支持 中 文 操作 ， 同 时 也 支 


持 英 文 输出 。 


11.1.2 ”功能 结构 


创建 PDF 文档 模块 的 功能 结构 如 图 11.1 文章 添加 


所 示 。 


11.1.3 ”程序 预览 PDF 文档 下 载 文章 浏览 


为 了 让 读者 更 好 地 了 解 创建 PDF 文档 模 图 11.1 PDF 文档 模块 的 功能 结构 图 
块 的 功能 ， 下 面 列 出 创建 PDF 文档 模块 中 一 
些 功 能 的 运行 效果 ， 供 广大 读者 参考 。 

文章 添加 页 面 的 运行 效果 如 图 11.2 所 示 。 
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图 11.2 文章 添加 页 面 的 运行 效果 
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文章 浏览 运行 效果 如 图 11.3 所 示 。 


网 上 企业 办 公 自 动 化 管理 系统 


Corporation OA Mi 


标题 作者 来 源 添加 时 间 
汪 宝 党 古人 古代 2011-05-19 00:41:46 
| 。 后 忆 由 言传 语 总 结 明日 科技 明日 科技 2011-05-19 00;37;40 
沐 宝 过 古人 古代 2011-05-19 00:41:46 
锭 程 词 由 言传 语 总 法 明日 科技 明日 科技 2011-05-19 00:37:40 
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内 部 邮件 管理 控 香 词 由 宣传 襄 总 结 明日 科技 明日 科技 2011-05-19 00:37:40 
邱 宝 读 古人 古代 2011-05-19 00:41:46 

ei 护理 让 富 全 名 法 明日 科技 明日 科技 2011-05-19 00:3T:40 

收 /发 文 管理 乓 宝洁 古人 古代 2011-05-19 00:41:46 
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图 11.3 文章 浏览 运行 效果 
文章 内 容 浏览 、PDF 文档 创建 和 下 载 链接 的 运行 效果 如 图 11.4 所 示 。 


ie 


更 新 时 间 : 2011-05-19 00:37:40 


| 
资产 管理 
地 
Ee PDF 文 档 的 创建 和 下 载 
| 退出 
ga gt tod sil re ny 
opr 如 1 CopyRights 5 reservad 2007 吉林 省 明日 科技 有 限 公司 


图 11.4 文章 内 容 浏览 、PDF 文档 创建 和 下 载 链接 的 运行 效果 


PDF 文档 下 载 的 运行 效果 如 图 11.5 所 示 。 
通过 Adobe Reader 阅读 PDF 文档 的 内 容 的 运行 效果 如 图 11.6 所 示 。 


@ 
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图 11.5 PDF 文档 下 载 的 运行 效果 图 11.6 浏览 PDF 文档 的 内 容 


11.2 数据库 设计 
在 本 模块 中 通过 MySQL 数据 库存 储 用 户 上 传 的 文章 , 然后 通过 PDF 格式 将 文章 内 容 下 载 到 本 地 。 


11.2.1 创建 数据 库 


在 本 模块 中 ， 创 建 db_pdf 数据 库存 储 用 户 上 传 的 文章 数据 。db_pdf 数据 库 中 只 包含 一 个 数据 表 ， 
即 tb_articles。 数 据 库 的 结构 及 数据 表 的 情况 如 图 11.7 所 示 。 


胃 服务 器 : localhost ， 轧 数据 库 : db_pdf 


表 - 操作 记录 数 ! ”类 型 整理 
了 arices 国 机 加 基 便 X 4 MySAM utf8_unicode_ci 
1 个 表 总 计 4 MylSAM uti8_general_ci 


图 11.7 db_pdf 数据库 结构 及 数据 表 的 情况 


11.2.2 ”创建 数据 表 


在 tb_articles 数据 表 中 存储 用 户 提 交 的 文章 信息 , 包括 id、 文章 标题 、 文 章 作者 、 来源、 添加 时 间 、 
文章 内 容 和 类 别 。tb_articles 数据 表 的 结构 如 图 11.8 所 示 。 


服务 器 : localhost ”局 教 据 库 : db_pdf 、 国 表 : th_articles 
字段 类 型 整理 属性 空 默认 酉 外 说 明 
过 int(10) 否 无 auto_increment ”了 
topic varchar(50) utfB_unicode_ci 否 无 文章 标题 
author varchar(50) utf_unicode_ci EE 文章 作者 
comefrom varchar(50) utfB_unicode _ci 否 无 来 源 
addtime timestamp 否 “CURRENT_ TIMESTAMP 添加 时 间 
content longtext utfB_unicode_ci 否 无 文章 内 容 
langver varchar(20) utf8_unicode_ci 否 无 类 别 


图 11.8 tb_articles 数据 表 的 结构 
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11.2.3 ”连接 数据 库 


本 模块 中 将 连接 数据 库 的 操作 存储 于 Connections\conn.php 文件 中 ， 设 置 数据 库 的 用 户 名 是 root， 
密码 是 111。conn.php 文件 的 代码 如 下 : 


<?php 

$conn=mysql_connect("localhost","root","111") or die(' 连 接 失败 . mysql_error()); /连接 服务 器 
mysql_select_db("db_pdf",$conn) or die (数据 库 选择 失败 :'. mysql_error()); // 选 择 数 据 库 
mysql_query("set names utf8"); // 设 置 页 面 编码 格式 
?> 


11.3 ”模块 国际 化 设置 


本 模块 的 国际 化 功能 只 是 一 个 辅助 ， 它 让 文章 的 添加 和 浏览 功能 支持 中 文 和 英文 两 种 语言 。 其 中 ， 
模块 国际 化 设置 的 步骤 如 下 : 

(1) 创建 lang 文件 夹 ， 编 写 lang_ch.php 和 lang_en.php 文件 ， 分 别 定义 中 文 语言 配置 文件 和 英文 
语言 配置 文件 ， 代 码 请 参考 光盘 中 的 内 容 。 

(2) 创建 config.php 文件 ， 获 取 浏 览 器 中 语言 选项 的 值 ， 根 据 语言 选项 的 值 调用 不 同 的 语言 配置 
文件 。 代 码 如 下 : 


<?php 

session_start(); 

$lang=$_SERVER['HTTP_ACCEPT_LANGUAGE'; /获取 语言 选项 
$lang=explode(';', $lang); /省 以 “;” 对 字符 串 进 行 分 隔 
$lang=$lang[0]; /为 数组 赋值 
S$lang=explode(',',$lang); / 咱 以 “,” 对 字符 串 进 行 分 隔 
$lang=$lang[0]; /为 数组 赋值 
switch($lang){ /根据 获取 到 的 语言 选项 的 值 ， 为 变量 赋值 
case "en-us": 

$lang="en"; 

break; 

case "zh-cn": 

S$lang="ch"; 

break; 

default : 

S$lang="en"; 

break; 

} 

iflisset($_SESSION[lang])){ 

$lang=$_SESSION[lang"; /为 Session 变量 赋值 


} 
include "lang/lang_S$lang.php"; // 调 用 不 同 的 语言 配置 文件 
?> 


[EN 
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(3) 创建 setlang.php 文件 ， 根 据 超 链接 中 传递 的 参数 ， 为 Session 变量 赋值 ， 并 且 调用 不 同 的 操 
作 页 面 。 代 码 如 下 : 


<?php 

session_start(); 

switch($_GET[lang'"]){ // 获 取 超 链接 传递 的 语言 选项 的 值 
case "en": 

$_SESSION[Iang]="en"; /根据 超 链 接 的 值 ， 为 Session 变量 赋值 
break; 

case "ch": 

$_SESSION['\lang"]="ch"; 

break; 

default : 

$_SESSION['lang]="en"; 

break; 


} 

if($_GETI'link]=="addnew"}{ // 根 据 超 链接 的 值 ， 跳 转 到 不 同 操作 页 面 
header("Location:add_data.php"); 

}elseif ($_GETI'ink]=="show"){ 

header("Location:look_over.php"); 

}else{ 

header("Location:index.php"); 

a 


至 此 ，PHP 国际 化 设置 完毕 ， 下 面 讲解 模块 具体 功能 的 开发 。 
11.4 文章 浏览 


11.4.1 文章 浏览 概述 


文章 浏览 功能 默认 输出 英文 格式 的 文章 标题 数据 ， 运 行 结 果 如 图 11.9 所 示 。 当 单 击 文章 标题 的 超 
链接 后 ， 将 浏览 到 文章 的 详细 信息 ， 并 且 在 详细 信息 页 还 设置 了 PDF 文档 下 载 的 超 链 接 ， 运 行 效果 如 
图 11.10 所 示 。 


图 11.9 英文 格式 的 文章 列表 


辐 转 因 ” PHP 典型 模块 开发 全 程 实录 


Chinese 
"English 


， Mankind need sore knowlsdge also constantly 


Under this £. 


Update Tine: 2011-05-19 02:35:21 


| ] Taxxsia 


二 Manpowernes 


X mt 
医 僵 风 反馈 | 类 助 中 心 | 网 站 地 图 | 招聘 英才 | 联系 我 们 
A 和 Ll CopyRiehts © reserved 2007 吉林 省 明日 科技 有 限 公司 
Manage System 


图 11.10 英文 格式 的 文章 浏览 
11.4.2 ”文章 列表 实现 过 程 
创建 index.php 文件 ， 根 据 选择 的 语言 不 同 ， 输 出 不 同 的 页 面 和 内 容 。 首 先 ， 通 过 require_once0 


语句 调用 不 同 的 包含 文件 ， 完 成 数据 库 连接 和 语言 配置 ， 然 后 ， 执 行 查 询 操作 ， 从 数据 表 中 读 取 出 指 
定 的 数据 信息 ; 最 后 ， 在 页 面 中 输出 数据 。 完 整 代码 如 下 : 


<?php 

require_once('Connections/conn.php'); // 包 含 数据 库 连接 文件 
require_once('config.php'); // 包 含 语言 配置 文件 
$query_rs="select * from tb_articles where langver='$rs filter order by id desc"; // 定 义 查 询 语句 
$rs=mysql_query($query_rs,$conn) or die(mysql_error()); // 执 行 查询 语句 
$row_rs=mysql_fetch_assoc(S$rs); // 获 取 查 询 结 果 

?> 

<html xmlns="http://www.w3.org/1999/xhtml"> 

<head> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<title><?php echo $index_title ?></title> 

<link href="css/style.css" rel="stylesheet" type="text/css" /> 

</head> 

<body class="twoColLiqLtHdr"> 

<div id="container"> 

<div id="header"><img src="images/<?php if($rs_fiter=="en"){fecho "OA_021.jpg";}else{echo "OA_02.jpg";}?>" 
width="780" height="120" border="0" usemap="#Map" /> 

<map name="Map" id="Map"><area shape="rect” coords="709,73,776,95" href="setlang.php?lang=ch&link= 
addnew"/> 

<area shape="rect" coords="706,98,772,113" href="setlang.php?lang=en&link=addnew" /></map> 

<!- end #header —></div> 

<div id="sidebar1"> 

<img src="images/<?php if($rs_filter=="en"}Mecho "OA_061.jpg";}else{echo "OA 06.jpg";}?>" 
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width="188" height="396" border="0" usemap="#Map2" /> 

<map name="Map2" id="Map2"><area shape="rect" coords="47,110,154,131" href="index.php" /> 
<area shape="rect" coords="43,71,163,98" href="add_data.php" /></map> 

<!~ end #sidebar1 —></div> 

<div id="mainContent"> 

<P><?php echo $index_topic; ?> 

<?php echo $index_author; ?> 

<?php echo $index_comefrom; ?> 

<?php echo $index_addtime; ?> 

</P> 


?><P> 

<a href="look_over.php?id=<?php echo $row_rs[id];?>"><?php echo $row_rs[topic];?></a> 
<?php echo $row_rs['author];?> 

<?php echo $row_rs['comefrom'];?> 

<?php echo $row_rs['addtime'];?> 

</P> 

<?php 

}while($row_rs=mysql_fetch_assoc($rs)); 

mysql_free_result($rs); 

?> 

<!- end #mainContent --></div> 

<div id="footer"><img src="images/OA_13-14.jpg" width="780" height="78" /> 
<!~ end #footer --></div> 

<!- end #container --></div> 

</body> 

</html> 


入 注 意 index.php 文件 中 输出 的 内 容 ， 会 根据 中 文 和 英文 不 同 语言 传递 的 参数 值 进行 切换 输出 。 


11.4.3 ”文章 浏览 实现 过 程 


当 用 户 单 击 文章 的 标题 时 ， 跳 转 到 look_over.php 文件 中 , 在 该 文件 中 根据 超 链接 传递 的 文章 id 查 
询 出 该 文章 的 详细 信息 ， 并 输出 文章 的 内 容 。 同 时 在 look_over.php 文件 中 创建 “下 载 PDF 文档 ” 超 链 
接 ， 链 接 到 setup_pdfphp， 在 该 文件 中 完成 PDF 文档 的 生成 和 下 载 操 作 。look_over.php 文件 的 关键 代 
人 码 如 下 : 


<?php 

require_once('Connections/conn.php'); // 连 接 数 据 库 
require_once('config.php'); // 配 置 语言 
$colname_rs='1'; // 定 义 变量 值 
ifisset($_GET[id])) // 检 测 变量 是 否 被 设置 
$colname_rs=(get_magic_ quotes_gpc())?$_GET[id]:addslashes($_GET[id]); // 对 数据 格式 进行 转换 
} 

$query_rs=sprintf("select * from tb_articles where id=%s",$colname _rs); // 对 字符 串 进行 格式 化 


辐 
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$rs=mysql_query($query_rs,$conn) or die(mysql_error()); /| 执行 查询 操作 
$row_rs=mysql_fetch_array($rs); /获取 查询 结果 集 
<html xmlns="http:/www.w3.org/1999/xhtml"> 

<head> 


<meta http-equiv="Content-Type" content= "text/html; charset=utf-8" /> 

<title><?php echo S$index_title ?></title> 

<link href="css/style.css" rel="stylesheet" type="text/css" /> 

</head> 

<body class="twoColLiqLtHdr"> 

<div id="container"> 

<div id="header"><img src="images/<?php if($rs_filter=="en"}{echo "OA_021.jpg";}else{echo "OA_02.jpg";}?>" 
width="780" height="120" border="0" usemap="#Map" /> 

<map name="Map" id="Map"><area shape="rect” coords="708,73,775,95" href="setlang.php?lang=ch&link= 
addnew" /> 

<area shape="rect" coords="706,98,772,113" href="setlang.php?lang=en&link=addnew" /></map> 
<!- end #header —></div> 

<div id="sidebar1"> 

<img src="images/<?php if($rs_filter=="en"}{echo "OA_061.jpg";}else{echo "OA_06.jpg";}?>" 
width="188" height="396" border="0" usemap="#Map2" /> 

<map name="Map2" id="Map2"><area shape="rect" coords="47,111,154,132" href="index.php" /> 
<area shape="rect" coords="43,71,163,98" href="add_data.php" /></map> 

<!- end #sidebar1 —></div> 

<div id="mainContent"> 

<P><?php echo $row_rs['topic'];?></p> 

<P><?php echo $show_author; ?>: <?php echo $row_rs['author];?></p> 

<P><?php echo $show_comefrom;?>: <?php echo $row_rs['comefrom'];?></p> 

<P><?php 

S$len=iconv_strlen($row_rs['content],'utf-8"); // 对 文章 内 容 长 度 进行 统计 
if($len>100){ 

echo iconv_substr($row_rs['content],0,100,utf-8). // 对 文章 内 容 进 行 截 取 输 出 
}else{ 

echo $row_rs['content]; // 输 出 文章 内 容 


?></p> 

<P><?php echo $show_upttime;?>: <?php echo $row_rs['addtime'];?></p> 
<P><a href="setup_pdf.php?id=<?php echo $row_rs['id]; ?>" 
target="_blank"><?php echo $show_down;?></a></p> 

<!~— end #mainContent --></div> 

<div id="footer"><img src="images/OA_13-14.jpg" width="780" height="78" /> 
<!— end #footer --></div> 

<!~— end #container --></div> 

</body> 

</html> 

<?php 

mysql_free_result($rs); 

?> 


@ 
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[a 技巧 在 对 文章 内 容 进行 统计 和 截取 时 ， 应 用 的 是 iconv *# 系 列 函 数 ， 通 过 iconv_strlen(0) 函 数 统 
计 文 章 的 长 度 ， 通 过 iconv_substr() 函 数 对 文章 内 容 进 行 截取 。 使 用 iconv * 系 列 函 数 对 字符 串 进行 
操作 时 ， 可 以 避免 截取 中 文字 符 串 出 现 乱码 的 问题 。 如 果 这 里 应 用 strlen() 函 数 统计 文章 长 度 ， 通 过 
substr0 函 数 对 文章 内 容 进 行 截取 ， 那 么 截取 的 内 容 可 能 会 出 现 乱码 。 


11.4.4 ”PDF 文档 创建 和 下 载 


PDF 文档 的 下 载 应 用 的 是 超 链接 下 载 方式 , 而 PDF 文档 的 创建 在 setup_pdf php 文件 中 完成 ， 其 根 
据 超 链接 传递 的 id 值 ,从 数据 库 中 查询 出 指定 文章 的 详细 信息 ,应 用 PDF_Chinese 类 中 的 方法 完成 PDF 
文档 的 创建 。setup_pdfphp 文件 的 完整 代码 如 下 : 


<?php 

header("Content-Type:text/html;charset=gb2312"); // 设 置 页 码 编码 格式 
include(pdfrchinese.php'); // 载 入 中 文 组 件 
include(Connections/conn.php'); // 载 入 数据 库 连接 文件 
class PDF extends PDF_Chinesef /继承 中 文 类 
function Header()f /定义 方法 设置 标题 
Sthis->SetFont(gb,"10): // 设 置 字体 
Sthis->Write(10, 文 章 '); /| 输出 标题 内 容 
S$this->Ln(20); /| 换行 

} 

function Footer(X{ // 设 置 页 脚 


$this->SetY(-15); 
$this->SetFont('gb,",10); 
$this->Cell(0,10,' 第 '.$this->PageNo().' 页 '); 
} 


} 
$colname_rs_id=$_GET[id]; 


// 获 取 指 定数 据 的 id 


$query_rs_article=sprintf("select * from tb_articles where id=%s",$colname_rs_id); /定义 SQL 语句 


$rs_article=mysql_query($query_rs_article,$conn) or die(mysql_error()); // 执 行 SQL 查询 语句 
$row_rs_article=mysql_fetch_assoc($rs_article); // 获 取 查 询 结果 

$pdf=new PDF(); /实例 化 PDF 类 

$pdf->AddGBFont(); // 设 置 字体 

$pdf->Open(); 1/ 打开 文件 

$pdf->AliasNbPages(); /为 每 页 定义 一 个 别名 

$pdf->AddPage(); /分 页 方法 

$pdf->SetFont(gb',B',20); // 设 置 字体 


$pdf->Cell(0,10,iconv("utf-8","gb2312", $row_rs_article['topic"])); 


$pdf->Ln(); 
$pdf->SetFont('gb',",10); 


// 输 出 一 个 内 容 
/| 换行 


$pdf->Cell(0,10,iconv("utf8","gb2312",$row_rs_article[' author])); 


$pdf->Ln(); 
$pdf->SetFont(gb',",10); 


$content= iconv(utf-8,gb2312',$row_rs_article[ content]); 


$pdf->MultiCell(0,5,$content); 


/设置 编码 格式 
// 输 出 内 容 ， 并 且 执 行 终止 


@ 
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$pdf>Ln(); 

$pdf->AddPage(); 1/ 分 页 
S$pdf->Output(iconv("utf-8","gb2312", $row_rs_article['topic”]).".pdf ,true); 。 // 生 成 PDF 文件 
$5 


Ped 
二 有 关 通 过 FPDF 创建 PDF 文档 的 详细 讲解 请 参考 本 章 的 11.6 节 。 


11.5 文章 添加 


11.5.1 文章 添加 概述 


文章 添加 功能 将 用 户 提交 的 文章 数据 添加 到 指定 的 数据 表 中 。 英 文 格式 文章 添加 的 运行 效果 如 图 11.11 
所 示 。 
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图 11.11 英文 格式 文章 添加 的 运行 效果 
11.5.2 ”文章 添加 实现 过 程 


文章 添加 操作 在 add_data.php 文件 中 完成 。 可 以 将 其 划分 为 两 部 分 : 第 一 部 分 是 通过 表单 提交 文 
章 数 据 ; 第 二 部 分 是 获取 表单 提交 的 数据 ， 将 其 添加 到 指定 的 数据 表 中 。 
第 一 部 分 创建 表单 的 代码 如 下 : 


<form name="form" method="post" action=""> 

<p><?php echo $add_data_topic; ?>: <input type="text" name="topic" value="" /></p> 
<p><?php echo $add_data_author; ?>: <input type="text" name="author" value="™" /></p> 
<p><?php echo $add_data_comefrom; ?>: <input type="text" name="comefrom" value="" /></p> 
<p><?php echo $add_data_content; ?>: </p> 

<textarea name="content" id="textarea" cols="65" rows="6"></textarea> 

<p><?php echo $add_data_langsp; ?>: 
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<input type="radio" name="langver" value="en" />&nbsp;&nbsp;<?php echo $add_data_lenver ?>: 
<input type="radio" name="langver" value="ch" /><?php echo $add_data_Ichver ?></p> 

<p><input type="submit" name="Submit" value="<?php echo $add_data_submit ?>" />&nbsp;&nbsp; 
<input type="reset" name="Reset" value="<?php echo $add_data_reset ?>" /></p> 

</form> 


第 二 部 分 的 代码 如 下 : 


<?php 


header("Content-type:text/html;charset=utf-8"); // 设 置 页 面 编码 格式 
require_once('Connections/conn.php'); // 连 接 数据 库 
require_once('config.php'); /配置 语言 

/定义 格式 转换 的 方法 


function GetSQLValueString($theValue,$theType,$theDefinedValue="",$theNotDefinedValue=""}{ 
S$theValue=(!get_magic_quotes_gpc()) ? addslashes($theValue):$theValue; 

switch($theType){ 

case "text"; 

S$theValue=($theValue !="")? "". $theValue. "" : "NULL"; 

break; 

} 

return $theValue; 

} 

if(isset($_POST[ Submit])X 

1/ 获取 表单 中 提交 的 数据 ， 并 完成 对 数据 格式 的 转换 

S$insertSQL=sprintf("insert into tb_articles(topic,author,comefrom,content,langver) values(%s,%s,%s,%s,%s)"， 
GetSQLValueString($_POST[topic],"text )， 

GetSQLValueString($_POST[author],"text")， 

GetSQLValueString($_POST['comefrom',"text"), 

GetSQLValueString($_POST[content'],"text"), 

GetSQLValueString($_POST[langver],"text") 


); 

// 执 行 添加 语句 ， 将 数据 添加 到 数据 表 中 

$Result1=mysql_query($insertSQL,$conn) or die(mysql_error()); 

echo "<script>alert('$add_data_Suc); window.location.href='add_data.php';</script>"; 
} 


?> 


-/. 
后 本 模块 中 的 功能 偏 少 。 因 为 本 模块 的 重点 是 讲解 PDF 文档 的 创建 ， 所 以 这 里 只 选择 文章 
浏览 和 文章 添加 作为 辅助 内 容 。 如 果 读者 感 兴趣 ， 可 以 在 本 模块 基础 上 进行 二 次 开发 ， 增 加 更 多 更 
好 的 功能 。 


11.6 技术 提炼 


在 11.4.4 节 中 讲解 的 PDF 文档 的 创建 和 下 载 , 只 是 从 整个 模块 的 角度 出 发 , 简单 地 给 出 了 PDF 文 
档 创建 和 下 载 所 用 的 文件 ， 以 及 程序 代码 。 下 面 将 对 FPDF 类 中 的 方法 进行 系统 讲解 。 
_ 国 
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11.6.1 FPDF 概述 


FPDF 是 一 个 免费 的 PDF 操作 类 ， 通 过 它 可 以 完成 基本 的 PDF 创建 功能 ， 并 且 它 还 支持 中 文 ( 需 
要 对 相应 字体 进行 配置 ) 。 FPDF 类 中 常用 的 PDF 操作 函数 包括 Open0 函 数 、AddPage0 函 数 ，SetFontO 
函数 、Cell0 函 数 和 Output(O) 函 数 。 

FPDF 的 下 载 可 以 通过 访问 http:/www.fpdforg/ 来 得 到 , 将 下 载 的 文件 直接 解压 到 某 一 文件 夹 即 可 。 


11.6.2 ”创建 简单 的 PDF 文档 


通过 FPDF 创建 PDF 文档 , 首先 要 实例 化 FPDF 类 , 然后 调用 类 中 的 函数 (Open0 函 数 、AddPage() 
函数 、SetFont() 函 数 、Cell() 函 数 和 Output() 函 数 ) 完成 文档 的 创建 。 
(1) 创建 FPDF 对 象 的 语法 格式 如 下 : 


S$pdf = new FPDF([string page-orientation [, string measure-unit [, string page-format]]]) 
FPDF 对 象 的 参数 说 明 如 表 11.1 所 示 。 
表 11.1 FPDF 对 象 的 参数 说 明 


age-orientation 


measure-unit 


设置 PDF 文档 纸张 的 类 型 。 可 选 值 为 : 表示 纸张 类 型 的 字符 串 ， 例 如“A4”、“A5”、“Letter”， 
或 者 一 个 包含 有 两 个 元 素 的 二 维 数组 ， 直 接 指明 纸张 的 大 小 


(2) Open0 函 数 用 于 标识 开始 创建 PDF 文档 ， 其 语法 格式 如 下 : 
$pdf->Open() 

(3) AddPage() 函 数 为 PDF 文档 添加 一 个 新 页 ， 其 语法 格式 如 下 : 
$pdf ->AddPage([string page-orientation]) 


其 中 ，page-orientation 表示 创建 的 PDF 文档 是 横向 还 是 竖 向 的 。 与 前 面 介 绍 的 创建 对 象 时 的 使 用 
方法 相同 。 
(4) SetFontO 函 数 设置 当前 使 用 的 字体 ， 其 语法 格式 如 下 : 
$pdf->SetFont(string font [, string style [, float size]])); 


其 中 ，font 表示 字体 。style 表示 样式 ，style 的 可 选 值 有 如 下 3 种 。 如 果 没 有 指定 style 的 值 ， 则 默 
认为 普通 的 形式 。 


page-format 


B: 粗 体 。 
I: 斜体 。 
U: 下 滑 线 。 


size 表 示 字 体 的 大 小 。 如 果 不 指定 则 取 默 认 值 ， 为 12pt。 
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(5) Cell0 函 数 用 于 为 当前 PDF 文档 增加 一 个 单元 格 。 其 语法 格式 如 下 : 


$pdf->Cell(float w [, float h [, string txt [, mixed border [, int In [, string align [, int fill [, mixed link]]J) 


Cell0) 函 数 的 参数 说 明 如 表 11.2 所 示 。 
表 11.2 Cell() 函 数 的 参数 说 明 


参数 说 明 
w 表示 单元 格 宽度 
h 表示 单元 格 高 度 
txt 表示 单元 格 中 添加 的 文本 字符 串 


表示 单元 格 的 边框 。 设 置 为 0 表示 无 边框 ;设置 为 1 表示 边框 。 或 者 表示 左边 框 ; T 表 示 顶 部 边框 ; 

R 表 示 右 边框 ，B 表示 底部 边框 

全 表示 换行 后 ， 设 置 目 前 的 位 置 。 可 选 值 为 : 0 表示 向 右 移动 ; 1 表示 定位 到 下 一 行 的 开始 ，2 表示 向 下 
移动 。 默 认 值 为 0 


align 设置 字符 串 的 对 齐 方式 。 可 选 值 为 : 工 表示 左 对 齐 ; C 表示 居中 ; R 表示 右 对 齐 。 默 认 值 为 L 


fill 设置 单元 格 的 背景 。 可 选 值 为 ，0 表示 透明 ;1 表示 不 透明 。 默 认 值 为 0 
link 设置 一 个 URL 链接 或 者 由 AddLink() 函 数 添 加 的 内 部 链接 


(6) Output0 函 数 用 于 输出 PDF 文档 ， 其 语法 格式 如 下 : 


$pdf->Output([string filename [, bool download]]) 


其 中 ，filename 表示 要 存储 的 文件 名 。 如 果 不 指定 文件 名 ， 则 浏览 时 会 在 正 浏览 器 中 直接 打开 。 
download 表示 是 否 一 定 要 用 户 下 载 查 看 ， 默 认为 false。 
下 面 应 用 上 述 介 绍 的 函数 , 创建 一 个 PDF 文档 ,输出 一 段 英文 字符 串 。examplel.php 的 代码 如 下 : 


<?php 

define(FPDF_FONTPATH'font/; /定义 font 文件 夹 所 在 路 径 

require_once('pdf/fpdf.php'); // 包 含 FPDF 类 库 文 件 

$pdf=new FPDF('P', ,mm', 'A4'); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 
$pdf->Open(); // 开 始 创建 PDF 

$pdf->AddPage(); /增加 一 页 

$pdf->SetFont(‘Courier','l',20); // 设 置 字体 样式 

$pdf->Cell(0,0,'mrsoft!'); // 增 加 一 个 单元 格 

S$pdf->Output(); // 输 出 PDF 到 浏览 器 

?> 


运行 结果 如 图 11.12 所 示 。 
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图 11.12 ”简单 的 PDF 文档 
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[5 上 面 的 示例 是 将 PDF 文档 的 内 容 在 下 浏览 器 中 输出 ， 您 可 以 将 上 述 代码 中 的 
“$pdf->Output();” 修 改 为 “$pdf->Output('mrsoft.pdf,true);”， 表 示 定 义 PDF 文档 名 称 为 mrsoft.pdf， 
设置 用 户 下 载 查 看 。 这 样 ， 就 完成 了 将 PDF 文档 保存 到 服务 器 并 提示 用 户 下 载 的 操作 。 


11.6.3 在 PDF 中 插入 图 片 


在 FPDF 中 通过 Image() 函 数 向 PDF 中 插入 图 片 ， 其 语法 格式 如 下 : 
Image(string file, float x, float y, float width, float height) 


其 中 ，x 和 yy 表示 图 片 所 在 的 坐标 ; width 和 height 表示 图 片 的 宽度 和 高 度 。 
下 面 应 用 Image() 函 数 向 PDF 文档 中 插入 一 张 图 片 ， 并 且 将 PDF 在 正 浏览 器 中 直接 输出 。 
example2.php 的 代码 如 下 : 


<?php 

define('FPDF_FONTPATH','font/’); // 定 义 font 文 件 夹 所 在 路 径 

require_once('pdf/fpdf.php'); // 包 含 FPDF 类 库 文件 

S$pdf=new FPDF('P', mm,', 'A4'); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 
$pdf->Open(); /创建 PDF 

$pdf->AddPage(); /增加 一 页 

$pdf->SetFont(Courier,,20); /设置 字体 样式 

$pdf->Image('images/as.jpg',10,10,80,80); /| 插入 图 片 

$pdf->Output(); /| 输出 PDF 到 浏览 器 

?> 


其 运行 效果 如 图 11.13 所 示 。 


镍 http//192.168.1.96/TM/pdf/pdi/example2.phF - Windows Interne.. eal El ES 
OO |e rapezlealeerwpdypdf ~ [5+ [x |[P am Pp-| 


| 次 eB | 古 apy1197168196FTM/pdWpdWexampieIPhp 


Cal | 


11.13 在 PDF 中 插入 图 片 


11.6.4 设置 PDF 文档 的 页 眉 与 页 脚 


在 FPDF 中 ,页 眉 与 页 脚 是 通过 重 写 FPDF 类 中 的 Header() 方 法 和 Footer() 方 法 来 实现 的 。 因 为 FPDF 
类 中 的 Header() 方 法 和 Footer() 方 法 虽然 存在 , 但 是 方法 体 中 没有 任何 内 容 。 所 以 , 设置 PDF 文档 的 页 
眉 与 页 脚 需 要 重 写 FPDF 类 中 的 Header( 方 法 和 Footer() 方 法 ， 完 成 页 眉 与 页 脚 的 设置 。 

下 面 创建 PDF 类 ， 继 承 FPDF 类 ， 在 PDF 类 中 对 Header 方法 和 Footer 方法 进行 重 写 ， 完 成 页 眉 


全 


入 
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与 页 脚 的 设置 。 然 后 创建 PDF 文档 ， 输 出 一 行 英文 字符 串 和 一 张 图 片 。example3.php 的 代码 如 下 : 


<?php 
define('FPDF_FONTPATH','font/’); 
require_once('pdf/fpdf.php'); 

class PDF extends FPDF{ 
function Header(X{ 
S$this->SetFont('Arial','B',15); 
S$this->Write(10,'Title'); 
S$this->Ln(20); 


function Footer(X{ 

$this->SetY(-15); 
S$this->SetFont('Arial','l',8); 

S$this->Cell(0, 10,'Page - '.$this->PageNo()); 
x 


} 

S$pdf=new PDF('P', ‘mm', 'A4'); 
$pdf->Open(); 

$pdf->AddPage(); 
$pdf->SetFont(‘Courier,'l',20); 
$pdf->Cell(30,30,MRSOFT1); 
$pdf->Image(images/as.jpg',30,50,80,80); 
$pdf->Output(); 


?5 


/定义 font 文件 夹 所 在 路 径 

// 包 含 FPDF 类 库 文件 

/| 继承 FPDF 类 ， 重 写 页 眉 与 页 脚 的 方法 
/设置 页 眉 

/设置 页 收 字 体 

// 写 入 页 收文 字 

/| 换行 


// 设 置 页 脚 
/设置 页 脚 所 在 位 置 
/设置 页 脚 字体 

/| 输出 当前 页 码 作为 页 脚 内 容 


/| 创建 新 的 FPDF 对 象 ， 竖 向 放 纸 ， 
/创建 PDF 

/增加 一 页 

/设置 字体 样式 

/| 输出 字符 串 

/| 插入 图 片 

/| 输出 PDF 到 浏览 器 


单位 为 毫米 ， 纸 张大 小 为 A4 


其 运行 结果 如 图 11.14 所 示 。 


| hespy/192168 .087Mp ap RjexampleI hp Wndows ntormet Explorer ese) 
fe, rr CEIECRG 
|[ 斌 em |Bhepy/192158156/TM/pdpdl /eampiea php 
[Ges 
: 
Title— SD> 
MRSOFT! wp 
4 
一 、 < 和 >” 
( 了 ) ss 
一 
Paoe-1 
ET 


图 11.14 设置 PDF 的 页 眉 和 页 脚 
在 设置 PDF 的 页 眉 与 页 脚 时 ， 引 入 了 Write()、Ln()、PageNo() 和 SetY() 4 个 方法 。 
Write() 方 法 用 于 输出 字符 串 。 当 到 达 文 档 的 右边 位 置 或 者 遇 到 (\n) 时 ， 会 自动 换行 。 其 语法 格式 
如 下 : 


Write(float h, string txt [, mixed link]) 
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参数 h 定 义 字符 串 的 行列 高 度 ;参数 txt 指 定 输出 的 字符 串 ;参数 link 设 置 链接 的 网 页 或 者 AddLink0 
方法 的 标识 符 。 
Ln() 方 法 用 于 换行 操作 ， 其 语法 格式 如 下 : 


Ln([float h]) 

参数 h 用 于 设置 行 的 高 度 。 默 认 值 为 最 后 输出 行 的 高 度 。 

PageNo() 方 法 的 作用 是 返回 当前 页 码 ， 其 语法 格式 如 下 : 

int PageNo(); 

SetY() 方 法 用 于 设置 页 脚 的 坐标 。 其 语法 格式 如 下 : 

SetY(float y); 

参数 y 为 页 面 上 的 Y 坐标 ， 单 位 为 毫米 。 如 果 y 为 负数 ， 则 表示 从 页 面 底部 向 上 的 距离 。 例 如 ， 
上 面 的 示例 中 设置 y 的 值 为 -15， 即 页 脚 所 在 位 置 距 页 面 底 部 15 上 毫米。 
11.6.5 在 PDF 文档 中 绘制 表格 


通过 FPDF 绘制 表格 的 方法 与 输出 文字 的 方法 类 似 , 在 输出 文字 时 往往 指定 Cell0 方 法 的 边框 参数 
为 0， 即 不 输出 边框 。 在 绘制 表格 时 往往 将 其 设置 成 大 于 0 的 整数 ， 用 于 表示 边框 ， 这 就 是 PDF 文档 
中 表格 的 单元 格 。 当 我 们 把 这 些 单 元 格 组 合 到 一 起 ， 就 构成 了 表格 。 

下 面 看 一 个 输出 表格 的 示例 ，example4.php 的 代码 如 下 : 


<?php 

define(FPDF_FONTPATH'font/; /定义 font 文件 夹 所 在 路 径 
require_once('pdf/fpdf.php'); // 包 含 FPDF 类 库 文件 

class PDF extends FPDF{ 

function Header()f // 设 置 页 眉 
Sthis->SetFont('Arial','B',15); // 设 置 页 眉 字 体 
Sthis->Write(10,Pay Slip'); // 写 入 页 眉 文字 
Sthis->Ln(20); /| 换行 

} 

function Footer()f // 设 置 页 肢 

Sthis->SetY(-15); /设置 页 脚 所 在 位 置 
Sthis->SetFont(Arial,'l',8); // 设 置 页 脚 字体 
$this->Cell(0,10,'Page - ".$this->PageNo()); 。 // 输 出 当前 页 码 作为 页 脚 内 容 
} 

} 

$pdf=new PDF(P', ‘mm', 'A4'); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 毫米 ， 纸 张大 小 为 A4 
$pdf->Open(); /开始 创建 PDF 
$pdf->AddPage(); /增加 一 页 
$pdf->SetFont(Arial,",14); /设置 字体 样式 


S$header=array('Name',Age','Duty','Laborage’); 


$data=array(); 


/设置 表 头 
/设置 表 体 


$data[0] = array(Simon' 
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"employee'5,000.00); 


$data[1] = array('Elaine','25",'Manager','6,000.00"); 


$data[2] = array(Susan' 


'25','employee','4,000.00"); 


$data[3] = array('David','26"','Manager,'8,000.00"); 


S$width=array(40,40,40,40); 
for($i=0;$i<count($header);$i++) 
S$pdf->Cell($width[$i],6,$header[$i],1); 
$pdf->Ln(); 

foreach($data as $row){ 
$pdf->Cell($width[0],6,$row[0],1); 
$pdf->Cell($width[1],6,$row[1],1); 
$pdf->Cell($width[2],6,$row[2],1); 
$pdf->Cell($width[3],6,$row[3],1); 
$pdf->Ln(); 


} 
$pdf->Output(); 
> 


运行 结果 如 图 11.15 所 示 。 


// 设 置 每 列 宽度 
/循环 输出 表 头 


/循环 输 出 表 体 


/输出 PDF 到 浏览 器 


@ http://192.168.1.96/TM/pdf/pdf/example4.php - Windows Internet Explorer 
GO |e) mpwaszlkaleerwpdtpdteompkeaphp -1s+[x|[P sng pr| 
| 请 收藏 赤 。 世 http://192.168.1.96/TM/pdf/pdf/example4.php 
ESEISIGIGAISIESEGTITEE IE 

| 器 Pay Slip 上 
|Z 
| Name Age Duty Laborage 
| 本 Simon 24 lemployee 5.000.00 
| Elaine 25 Manager 6.000.00 

Susan 25 lemployee 4,000.00 

David 26 Manager 8.000.00 

20998x29697 厘米 亚 


图 11.15 在 PDF 中 绘制 表格 


11.6.6 在 PDF 文档 中 输出 中 文 


如 果 应 用 前 面 讲解 的 技术 ， 直 接 通过 FPDF 在 PDF 文档 中 输出 中 文 ， 那 么 输出 的 中 文字 符 串 将 是 
乱码 。 解 决 此 问题 的 方法 是 应 用 FPDF 提供 的 一 个 中 文 插件 ,继承 插件 中 的 PDF_Chinese 类 , 在 PDF 


文档 中 输出 中 文 。 


首先 ， 下 载 并 解压 此 插件 ， 复 制 其 中 的 chinese.php 文件 ， 将 其 存储 于 项 目的 指定 文件 夹 下 ; 然后 
通过 include 语句 在 创建 PDF 文档 的 文件 中 载 入 chinese.php; 最 后 ， 创 建 PDF 类 ， 继 承 PDF_Chinese 
类 ， 在 PDF 文档 中 输出 中 文 。 示 例 example5.php 的 代码 如 下 : 


<?php 
include('pdf/chinese.php'); 
class PDF extends PDF_Chinese{ 


// 载 入 中 文 插件 
/| 继承 中 文 插件 类 
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function Header(X{ 
S$this->SetFont('GB',",10); 
S$this->Write(10,'FPDF 中 文 测试 ); 
S$this->Ln(20); 

} 

function Footer(){ 

$this->SetY(-15); 
$this->SetFont('GB',",10); 
$this->Cell(0,10,' 第 '.$this->PageNo().' 页 '); 
3 


} 

$pdf=new PDF(); 
S$pdf->AddGBFont(); 
$pdf->Open(); 
$pdf->AliasNbPages(); 
$pdf->AddPage(); 
S$pdf->SetFont(GB','l',20); 


/设置 页 丑 
/设置 字体 
/设置 页 眉 的 内 容 
/执行 换行 操作 


/ 设 定 页 脚 
/设置 页 脚 的 输出 坐标 
/设置 字体 
/设置 页 脚 输出 的 内 容 


/| 创建 PDF 文档 

/添加 字体 

/开启 文档 

/为 每 个 页 面 定义 一 个 别名 
/添加 页 

/设置 字体 


$str=' 山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 ,有 龙 则 灵 。 斯 是 陋室 ， 惟 吾 德 声 。 苔 痕 上 阶 绿 ， 草 色 入 廉 青 。 谈 笑 有 鸿儒 ， 
往来 无 白丁 。 可 以 调 素 琴 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 无 案 睦 之 劳 形 。 南 阳 诸 葛 访 ， 西 蜀 子 云 享 。 孔 子 云 ;: “ 何 陋 


之 有 ? ”"; 
$pdf->Write(10,$str); 
$pdf>Output(); 


> 


// 输 出 中 文 


其 运行 结果 如 图 11.16 所 示 。 


站 hatpy/1921681967TM/pdWpdfyexamglesphp Windows Intemet Explorer 


GO- | 四 http:/192.168.1.96/TM/pdf/pdflexample5.php 


SEE Bing dh 


这 收藏 交 


入 hutp://192.168.1.96/TM/pdi/pdf/example5.php 


同 罩 号 加 | 


jal Lx*[] | 


岂 不 在 疹 ， 育 仙 峙 名 。 水 不 在 座 ， 方 龙 出 灵 。 新 是 慑 室 ， 众 
吾 总 学 ， 考 疯 上 失学 ， 辣 色 人 谭 
了 。 可 以 尊 责 盏 ， 阅 企 郊 。 无 绎 f 
两 用 请 茵 访 ， 西 世子 去 党。 孔子 去 : 


谈 疾 方 况 伪 ， 苦 来 无 据 
乱 耳 ， 无 案 忆 之 劳 形 
有 县 之 育 ?“ 


图 11.16 在 PDF 中 输出 中 文 


人 (example5.php ) 时 ， 页 面 的 编码 格式 要 设置 成 gb2312， 如 
果 页 面 的 编码 格式 是 utf8， 那 么 即使 继承 了 PDF_Chinese 类 ， 输 出 的 中 文 仍旧 是 乱码 。 


11.6.7 PDF 文档 下 载 


要 实现 PDF 文档 的 下 载 操作 ， 只 需要 在 通过 Output0) 函 数 输出 PDF 文档 时 ， 添 加 两 个 参数 即 可 。 
第 一 个 参数 定义 下 载 的 PDF 文档 的 名 称 ， 第 二 个 参数 设置 为 ture， 提 醒 用 户 下 载 查 看 ， 如 果 将 第 二 个 


@ 
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参数 设置 为 包 lse， 那 么 生成 的 PDF 文档 会 直接 存储 于 服务 器 中 ， 不 会 提醒 用 户 下载 查 看 。 


所 下 


读 。 


可 以 


下 面 对 示 例 example5.php 进行 修改 ， 将 其 生成 的 PDF 文档 命名 为 “程序 测试 pdf”， 并 且 提 醒 用 


载 查看 。 修 改 的 关键 代码 如 下 : 

<?php 

include(pdfrchinese.php'); // 载 入 中 文 插件 
// 省 略 了 部 分 代码 


$str=' 山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 ,有 龙 则 灵 。 斯 是 陋室 ， 惟 吾 德 声 。 苔 痕 上 阶 绿 ， 草 色 入 廉 青 。 谈 笑 有 鸿儒 ， 
往来 无 和 白丁。 可 以 调 素 琴 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 无 案 息 之 劳 形 。 南 阳 诸 葛 庐 ， 西 蜀 子 云 亭 。 孔 子 云 ; “ 何 陋 
之 有 ?7 ” 

S$pdf->Write(10,$str); /| 输出 中 文 

$pdf->Output(' 程 序 测试 .pdf',true); 

£2 


将 修改 后 的 示例 存储 于 example6.php 文件 中 ， 运 行 此 示例 将 输出 如 图 11.17 所 示 的 PDF 文件 下 载 


Em 区 本 | 
您 想 打开 或 保存 此 文件 四 ? 
名 名 称 : 程序 出 试 .pdf 
> 类 型 ， Foxit Reader Document 4.82KB 


| 来 源 ，localhost 


OO 向 的 


图 11.17 PDF 文档 下 载 
11.7 本 章 小 结 


本 章 运用 FPDF 来 创建 文档 ， 通 过 创建 PDF 文档 ,浏览 者 可 以 通过 Adobe Reader 对 其 内 容 进行 阅 
同时 ， 在 创建 PDF 文档 的 过 程 中 ， 使 整个 系统 的 设计 思路 更 加 清晰 。 通 过 本 章 的 学 习 ， 读 者 不 仅 
了 解 PDF 文档 的 使 用 ， 而 且 可 以 进行 PDF 文档 下 载 。 


第 /2 这 


邮件 发 送 系统 


(PHP+Zend Mail 组 件 实现 ) 


电子 邮件 是 Internet 上 应 用 最 广 的 服务 ， 通 过 网 络 的 电子 邮件 系 
统 ， 用 户 可 以 用 极 低廉 的 价格 以 非常 快速 的 方式 与 世界 上 任何 一 个 角 
落 的 网 络 用 户 联络 ， 这 些 电子 邮件 可 以 是 文字 、 图 像 、 声 音 等 各 种 形 
式 。 电 子 邮 件 具 有 快速 、 便 接 、 低 成 本 等 优势 ， 适 合 网 络 上 的 信息 交 
流 。 在 构建 企业 站 点 时 ， 可 以 为 企业 用 户 提供 E-mail 通信 功能 ， 充 分 
利用 E-mail 的 功能 可 以 极 大 地 提高 企业 的 工作 效率 ,减轻 用 户 的 工作 
负担 。 电 子 邮 件 的 收发 技术 是 PHP 程序 员 必 须 掌握 的 技术 之 一 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

由 ”利用 mail() 国 数 发 送 电 子 邮 件 

WI ”利用 fsockopen() 国 数 接收 电子 邮件 


MW 带 附件 邮件 的 发 送 与 下 载 
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12.1 邮件 发 送 系统 概述 


随 着 Internet 技术 的 飞速 发 展 ， 越 来 越 多 的 人 选择 通过 网 络 来 进行 即时 沟通 。 本 章 讲解 了 邮件 发 送 
系统 从 登录 邮件 服务 器 ， 到 发 送 邮件 、 接 收 邮件 、 查 看 邮件 内 容 以 及 删除 邮件 等 一 系列 功能 的 实现 方 
法 。 其 中 还 详细 地 介绍 了 WinWebMail 的 安装 和 配置 方法 。 


12.1.1 电子 邮件 简介 


电子 邮件 (E-mail) 是 Intemet 上 最 广泛 的 应 用 之 一 。 用 户 必 须 拥 有 因特网 服务 商 (ISP) 提供 的 账 
户 或 信箱 ， 才 能 接收 因特网 的 信件 。 电 子 邮 件 用 途 十 分 广泛 ， 可 以 与 世界 各 地 的 朋友 和 客户 保持 联系 ， 
也 可 以 实现 收 /发 、 访 问 等 功能 。 

电子 邮件 地 址 的 格式 是 USER@SERVER.COM， 由 3 部 分 组 成 : 第 1 部 分 USER 代表 用 户 信箱 的 
账号 ， 对 于 同一 个 邮件 接收 服务 器 来 说 ， 该 账号 必须 是 唯一 的 ， 第 2 
部 分 “@” 是 分 隔 符 ; 第 3 部 分 SERVER.COM 是 用 户 信箱 的 邮件 接 
收服 务 器 域名 ， 用 以 标识 其 所 在 的 位 置 。 


12.1.2 ”功能 结构 


邮件 发 送 系统 的 所 有 信息 都 保存 在 POP3 服务 器 上 ， 所 以 不 涉及 
后台 管理 及 后 台数 据 库 。 邮 件 系统 的 功能 结构 如 图 12.1 所 示 。 


12.1.3 ”程序 预览 
为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 , 下面 给 出 本 系统 的 


儿 个 页 面 运行 效果 图 , 如 果 想 查看 完整 的 效果 图 , 请 参见 光盘 源 程序 。 
登录 邮件 服务 器 的 页 面 如 图 12.2 所 示 。 


器 


[i 
明日 科技 人 CR 
[EE 有] [ 重 下 ] 


图 12.2 登录 邮件 服务 器 的 页 面 
在 邮件 发 送 系 统 中 发 送 带 附件 邮件 的 页 面 运行 效果 如 图 12.3 所 示 。 
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由 hl 中 入: 关 达 缀 | 


RICE 


Er sort earsoft. con 
-rs 
-a 


[A Er 


的 大 草原 是 我 的 未 ， 那 里 丰 温 山 过 时 的 花 


12.3 发 送 带 附 件 邮件 的 页 面 运行 效果 
邮件 发 送 系统 中 收 件 箱 的 运行 效果 如 图 12.4 所 示 。 


发 件 人 时 间 | 大 小 
x 湿 333 | a | 482KB 
x 区 566 | a | 482KB 
Xx WT nrsoft@mrsoft.con Set 05 Jam203 | a. 55 
x 加 由 hrsoft@mrsoft. com St Si | 1175M 
x 便 忆 pineri@ningri. con Sat 人 | 544XB 
X 去 soFtemsoft ca im 07 Jan 2013 | 285. 1 


06:44:35 
您 的 邮箱 中 共有 邮件 : 6 考 占 用 空间 ; 1467I 


12.4 收 件 箱 的 运行 效果 
邮件 发 送 系统 中 查看 邮件 详细 信息 的 页 面 运行 效果 如 图 12.5 所 示 。 


@ 
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Non, O07 Jan 2013 06:44:35 志 
285. 4482421881 E 
风景 01. JPG 

美丽 的 大 草原 是 我 的 家 ， 那 里 有 漫 山 所 野 的 花 


图 12.5 查看 邮件 详细 信息 的 页 面 运行 效果 


12.2 ”邮件 收发 的 原理 


在 具体 讲解 邮件 发 送 系统 功能 之 前 ， 首 先 介绍 一 下 邮件 发 送 系 统 发 送 邮 件 、 接 收 邮件 、 查 找 邮 件 
及 删除 邮件 实现 的 基本 原理 。 
发 送 邮件 实现 相对 简单 ， 用 PHP 自身 提供 的 mailO 函 数 就 可 以 完成 该 功能 。 需 要 注意 的 是 ， 
由 于 mailO 函 数 自身 不 能 进行 SMTP 认证 ， 因 此 需要 安装 一 个 SMTP 服务 器 转发 邮件 ， 并 且 
需要 在 php.ini 中 设置 SMTP 服务 器 地 址 及 端口 号 。 
加 ”接收 邮件 需要 利用 fsockopen0) 函 数 来 实现 ， 首 先 利 用 该 函数 连接 POP3 服务 器 并 返回 一 个 
socket 句柄 , 之 后 就 可 以 利用 fputs() 函 数 将 POP 命令 发 送 给 服务 器 , 利用 fgets() 函 数 接收 POP 
服务 器 反馈 回来 的 信息 〈 如 邮件 头 、 邮 件 内 容 ) ， 从 邮件 头 中 就 可 以 提取 发 件 人 地 址 、 发 件 
时 间 等 信息 。 
查找 邮件 时 , 系统 首先 遍历 所 有 POP 服务 器 中 的 信件 , 并 提取 所 有 信件 的 主题 和 发 件 人 地 址 ， 
如 果 发 现 与 用 户 查 找 的 主题 或 发 件 人 地 址 相同 的 邮件 ， 就 将 该 邮件 显示 出 来 。 
删除 邮件 只 需 用 fputs0 函 数 将 POP 命令 DELE 写 入 服务 器 ， 但 需 注意 删除 邮件 必须 由 命令 
QUIT 来 执行 ， 否 则 即使 发 送 了 DELE 命令 也 无 法 完成 删除 工作 。 
以 上 内 容 为 整个 邮件 系统 实现 的 核心 思想 ， 笔 者 将 接收 邮件 的 各 项 功能 封装 到 类 POP3 中 ， 这 样 
做 的 目的 既 符 合 PHP 面向 对 象 的 发 展 趋势 ， 又 为 系统 带 来 了 很 好 的 移植 性 ， 并 且 为 日 后 的 维护 工作 带 


来 很 大 方便 。 
@ 
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12.3 通过 WinWebMail 配置 邮件 服务 器 


在 本 模块 中 ， 通 过 第 三 方 的 WinWebMail 邮件 服务 器 ， 在 本 地 测试 电子 邮件 系统 的 功能 。 下 面 介 
绍 一 下 WinWebMail 的 安装 和 配置 。 

(1) 打开 如 图 12.6 所 示 的 WinWebMail 3.7.7.1 标准 版 安装 界面 ， 单 击 Next 按钮 。 

(2) 进入 如 图 12.7 所 示 的 WinWebMail 3.7.7.1 标准 版 协议 界面 ， 单 击 Next 按钮 。 


沁 TinYeblail 3.7.7.1 标准 版 和 vobrail 3.7-7.1 标准 忒 


WinWebMail 软 件 产 品 是 
Welcome to WiiWebMal 3771 标准 版 Setup program This 区 人 
program wl install WinWebMal 377.1 标准 版 on your compuler 人 
四 


ltis shongl iecommended that you et al Windows programs 
before tunning this Setup Program 


WinWebM 加 软件 产品 最 终 用 户 许可 协议 


Cick Cancel to quit Setup and close any programs you have 
running Cck Ned to cninue wh the getup Progam 本 软件 产品 版 权 原 于 马 坚 所 有 ， 并 受 著作 权 法 、 国 际 版 权 公约 以 及 其 它 知识 产权 
之 法 律 及 条 约 之 保护 , 


本 『 软 件 产品 〗 仅 系 授权 使 用 。 


WARNING: This program is prolected by copyiight law and 
intemalional heabes 


Unauthorized reproduction or distribution of this program of any 


parion of i may result in severe cvi and cmnal penalies. and 攻 六 在 一 和 电 直上 安 交 信用 本 [软件 产品 。 
Wl be prosecuted to the maamum extent possible under law. 


2 、 限 制 。 
您 应 保留 所 有 『 软 件 产品 】 振 风 上 之 著作 权 标 示 。 


Wise Instalation Wzard? 


Cancel < Back Cancel 
图 12.6 ”WinWebMail 3.7.7.1 标准 版 安装 界面 图 12.7 WinWebMail 3.7.7.1 标准 版 协议 界面 


(3) 进入 如 图 12.8 所 示 的 WinWebMail 3.7.7.1 标准 版 安装 路 径 界面 ， 可 以 选择 所 要 安装 的 路 径 ， 
单 击 Next 按钮 。 


(4) 进 入 如 图 12.9 所 示 的 WinWebMail 3.7.7.1 标准 版 软件 名 称 界 面 , 单 击 Next 按 钮 进入 如 图 12.10 
所 示 的 安装 完毕 界面 ， 单 击 Finish 按钮 安装 完毕 。 


沁 WinYebail 3.7.7.1 标准 版 
Select Program Manager Group 


Destination Location 


Selup wilinstal WiniwebMail 37.7.1 标准 版 in the folowing fokder. 
To install into a diferent folder, click Browse, and select another folder 


Enter the name of the Program Manager group to add WiriWebMal 37.7.1 标准 版 icons to: 


You can choose nat to instal WinWebMal 37.7.1 标准 版 by cicking Cancel to ea Setup. Pe Server 

ev 
Mactomedis 
Microson Developer Network 
Microsoft 0fhce 工具 
Microsoft SQL Server 


Microsoft SQL Server 1 版 本 切换 
Destinaon Folder Mictosoft SQL Server 2005 
CAWinWebMail Browse. Microsoft Visual Studio 2008 
Microsoft Windows SDK YE OA 
Zend Studo for Echpse -6.1.1 日 


se Installation Wizard? wise Instalation Wizard? 


‘Bek [Cie] cae Back Caneal 
图 12.8 ”WinWebMail 3.7.7.1 标准 版 安装 路 径 界 面 图 12.9 WinWebMail 3.7.7.1 标准 版 软件 名 称 界 面 
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(5) 启动 WinWebMail, 用 户 可 以 单 击 右 下 角 带 红色 X 的 蓝 色 图 标 , 在 弹出 的 快捷 菜单 中 选择 “ 服 
务 ” 命 令 ， 如 图 12.11 所 示 。 


混 Winwebaail 3.7.7.1 标准 版 又] 


[7 WinWebMal 37.7.1 标 准 版 has been successiuly ietaled 
到 | Press the Finish button to ex this netalalion_ 


WinWebMai! Server 


图 12.10 WinWebMail 3.7.7.1 标准 版 安装 完成 界面 图 12.11 WinWebMail 选项 
(6) 当选 择 “ 服 务 ” 命 令 后 进入 如 图 12.12 所 示 的 服务 界面 ， 选 中 “修改 ” 复 选 框 ， 输 入 DNS 
的 IP 地 址 ， 单 击 “ 启 动 WinWebMail 服务 程序 ”按钮 ， 再 单 击 绿色 对 号 图 标 ， 服 务 选项 配置 完成 。 
(7) 重新 进入 图 12.11 所 示 的 操作 界面 ， 选 择 “域名 管理 ”命令 ， 弹 出 如 图 12.13 所 示 的 界面 。 
添加 域名 后 ， 单 击 绿色 对 号 图 标 。 
WinWebMail 服务 


DNS 设置 
习 修改 
首选 DNS 的 IF 地 址 
备用 DNS 的 IF 地 址 


人 当 DNS 查 询 MX 记 录 失 败 ， 从 DNS 根 服务 器 查询 设置 


ils 线 溃 保存 时 间 ， | 天 ”三 ] | 清空 所 有 DIS 仲 数 据 本 DXF# 
本 stem. Mail 
启动 设置 ee | 


启动 方式 : 5 普通 三 快速 


重新 读 取 邮箱 空间 使 用 数据 


WinNebllail 服 务 程序 
启动 WinWebllail 服 务 程序 
图 12.12 WinWebMail 服务 界面 图 12.13 WinWebMail 域名 管理 


(8) 重新 进入 图 12.11 所 示 的 界面 ， 选 择 “ 系 统 设置 ”命令 ， 弹 出 如 图 12.14 所 示 的 界面 。 添 加 
用 户 名 、 密 码 ， 选 择 所 在 域名 后 单 击 绿色 对 号 图 标 。 
至 此 ，WinWebMail 配置 完毕 。 
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Ocad51f6 


systen. nail 


站 FE EE 
用 户 名 : mrsoft a] SD: fe 域 : ESE =] 
注 释 : | 
厂 禁用 帐号 @) 厂 限制 外 发 ” 接 入 方式 : [http/sntp/pop3, inap4 | 
最 后 登录 日 期 [期 满 日 期 YYrmmD) :站 
[Ev] [mw]| | | 


® 


v x 


图 12.14 WinWebMail 系统 设置 


12.4 登录 邮件 服务 器 


用 户 登 录 邮件 服务 器 时 ， 在 如 图 12.15 所 示 的 登录 界面 中 输入 POP3 服务 器 地 址 、 用 户 在 POP3 服 
务 器 注册 的 邮箱 名 称 和 密码 ， 单 击 “ 登 录 ” 按 钮 ， 将 数据 提交 到 chklogin.php 文件 中 ， 完 成 与 服务 器 的 
连接 操作 。 


Eg localhost 
邮 箱 Prsoft@mrsoftcon 
明日 科技 密码 


EslE: 
12.15 ”邮件 服务 器 登录 


在 chklogin.php 文件 中 ， 获 取 表 单 提交 的 服务 器 地 址 、 邮 箱 名 称 和 密码 ， 调 用 POP3 类 中 的 open0 
和 login() 方 法 ， 完 成 与 服务 器 的 连接 ， 并 且 将 登录 成 功 后 的 服务 器 地 址 、 邮 箱 名 称 和 密码 赋 给 指定 的 
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Session 变量 。 关 键 代码 如 下 : 


<?php 

session_start(); 

include("mailclass.php"); // 包 含 发 邮件 类 

$hostname=$_POST['servemame'; /获取 提交 的 服务 器 地 址 
$mailname=substr($_POST[mailname',0,strppos($_POST['mailname'],'@')); 1/ 获取 提交 的 用 户 邮 箱 地 址 
$mailpwd=$_POST[mailpwd]; // 获 取 提 交 的 用 户 邮箱 密码 


S$rec=new pop3($hostname,110,10); // 对 发 邮件 类 进行 初始 化 

if(l$rec->open()){ ”// 如 果 调 用 该 类 的 open() 方 法 返回 false 值 ， 则 说 明 没有 成 功 连接 上 服务 器 ， 这 时 给 出 提示 
echo"<script>alert(' 对 不 起 ， 无 法 连接 服务 器 ! ');history.back();</script>"; 

exit; 


六 

1/ 如 果 调用 发 邮件 类 的 login() 方 法 返回 false 值 ， 则 说 明 用 户 输入 的 名 称 或 密码 错误 
if(!$rec->login($mailname,$mailpwd)X{ 

echo"<script>alert( 对 不 起 ， 您 的 用 户 名 或 密码 输入 错误 ! ');history.back();</script>"; 

exit; 

} 

$_SESSION[host]=$hostname;// 如 果 用 户 登录 成 功 ， 则 注册 Session 变量 host 保存 服务 器 地 址 ， 以 备 其 他 页 面 使 用 
// 如 果 用 户 登录 成 功 ， 则 注册 Session 变量 user 保存 用 户 邮箱 地 址 ， 以 备 其 他 页 面 使 用 
$_SESSION['user]=$_POST[mailname'; 

$_SESSION[pass]=$mailpwd;// 如 果 登 录 成 功 ， 则 注册 Session 变量 pass 保存 用 户 邮 箱 密码 ， 以 备 其 他 页 面 使 用 


S$rec->close(); // 调 用 发 邮件 类 的 close() 方 法 关闭 与 POP3 服务 器 的 连接 
header("location:list.php"); // 定 位 到 发 件 箱 页 面 
?> 


12.5 邮件 收发 


12.5.1 邮件 收发 模块 概述 


成 功 登录 邮件 服务 器 之 后 ， 就 可 以 完成 邮件 的 发 送 、 接 收 、 查 看 、 
查找 和 删除 操作 。 邮 件 收发 模块 的 功能 如 图 12.16 所 示 。 

在 发 件 箱 中 可 以 完成 无 附件 邮件 和 有 附件 邮件 的 发 送 操作 ， 在 收 
件 箱 中 可 以 查看 接收 到 的 邮件 主题 ， 以 及 邮件 的 具体 内 容 ， 同 时 还 可 
以 完成 附件 的 下 载 操作 ， 在 查找 邮件 中 可 以 根据 邮件 主题 和 发 件 人 对 图 12.16 邮件 收发 模块 的 功能 
邮件 进行 查找 ; 刷新 功能 完成 对 页 面 的 刷新 操作 ， 以 此 更 新 出 邮箱 最 
新 收 到 的 邮件 信息 ; 退出 实现 当前 登录 用 户 的 退出 操作 。 


12.5.2 ”发 送 邮件 


发 送 邮件 的 操作 由 listphp 和 sendmail.php 两 个 文件 完成 。 在 listphp 中 定义 发 送 邮 件 表单 ， 如 图 12.17 
所 示 。 
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发 件 人 :|adnin@systen. nail 

由 件 人 : Jadnin@systen. nail 

3 

密 送 : | 

EL 

阳 件 玉 :\Users\Adninistrator\Desktop\jQueryAPI” 浏览... 
程序 测试 ,发送 带 附件 的 邮件 


内 容 : 


[重要 | 
图 12.17 发 送 邮 件 
为 了 缓解 服务 器 压力 ,在 添加 发 送 邮 件 的 信息 时 , 应 用 JavaScript 脚本 对 表单 提交 的 数据 进行 验证 ， 


其 中 发 件 人 、 收 件 人 和 邮件 内 容 为 必 填 内 容 ， 并 且 格 式 必须 正确 ， 而 附件 为 可 选项 ， 抄 送 和 密 送 不 必 
填写 。 验 证 表单 提交 数据 格式 是 否 正 确 使 用 的 是 JavaScript 脚本 的 自 定义 函数 chkinput()， 该 函数 的 部 
分 代码 如 下 : 

tliSt Dh rrr 

<script language="JavaScript"> 

function chkinput(form) /定义 chkinput() 函 数 ， 用 来 验证 用 户 输入 的 信息 

{ 

if(form.mailfrom.value=="") // 判 断 用 户 输入 的 信息 是 否 为 空 

{ 

// 如 果 用 户 输入 的 信息 为 空 ， 则 利用 window 的 alert() 方 法 弹出 一 个 提示 框 ， 告 诉 用 户 某 些 必 填 的 内 容 没有 填写 

alert(" 请 输入 发 件 人 地 址 "); 

form.mailfrom.select(); /| 选择 当前 窗 体 元 素 

return(false); /| 返回 false 值 使 表单 不 能 提交 

} 


/省 略 了 功能 类 似 的 代码 ， 请 见 本 书 附带 光盘 
/ 员 于 电子 邮箱 地 址 中 一 定 含有 字符 “@”， 所 以 通过 用 户 输入 的 邮箱 地 址 是 否 含有 字符 “@” 就 可 以 判断 用 户 输 
入 的 邮箱 地 址 是 否 正确 ， 函 函数 indexOf( 的 作用 是 判断 该 函数 参数 在 某 字符 率 中 第 一 次 出 现 的 位 置 ， 如 果 结果 小 于 
0， 则 说 明 没有 查找 到 该 字符 
if(form.mailfrom.value.indexOf('@')<0) 


{ 

alert(" 发 件 人 E-mail 格式 输入 错误 人"); 
form.mailfrom.select(); 

return(false); 


} 


return(true); // 如 果 用 户 输入 的 信息 通过 验证 ， 则 该 函数 返回 true， 从 而 使 表单 能 够 提交 


} 
</script> 
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在 sendmail.php 文件 中 ， 首 先 获取 表单 提交 的 邮件 内 容 。 然 后 对 邮件 标题 和 内 容 进 行 编码 格式 转 


。 最 后 根据 附件 的 内 容 进 行 判断 ， 如 果 附 件 为 空 ， 则 直接 应 用 mail0 函 数 完成 邮件 的 发 送 ; 如 果 附 件 
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不 为 室 ， 则 需要 通过 文件 系统 函数 读 取 附 件 的 内 容 ， 对 其 编码 格式 进行 转换 ， 定 义 到 邮件 体 中 后 ， 应 
用 mail0 函 数 完成 邮件 的 发 送 。 关 键 代码 如 下 : 


CIMertenaniennenennanernsanarnauS Ondail. ha 


<?php 

include("function.php"); 

$mailfrom=$_POST['mailfrom']; // 获 取 发 件 人 地 址 
$mailto=$_POST['mailto"]; // 获 取 收 件 人 地 址 
$mailcc=$_POST[mailcc']; // 获 取 抄 送 人 地 址 
$mailbcc=$_POST[mailbcc']; /获取 密 送 人 地 址 
$mailsubject=base64_encode($_POST[mailsubject]); /获取 邮件 主题 
$mailbody = base64_encode($_POST[mailbody]); /获取 邮件 内 容 


if(empty($_FILES['upload_ file][rname'])i{ 

$headers = "From:$mailfrom \n Cc:$mailcc \n Bcc:$mailbcc"; // 定 义 邮 件 头 
$headers .= "Reply-To:$mailfrom\nn"; 

$headers .= "Content-type: text/html;\An charset=iso-8859-1 \r\n"; 
if(@mail($mailto, $mailsubject,$mailbody,$headers)}{ // 通 过 mail() 函 数 发 送 邮件 ， 并 给 出 邮件 发 送 结果 
echo "<script>alert(' 邮 件 发 送 成 功 ! ');history.back()</script>"; 

}else{ 

echo "<script>alert(' 邮 件 发 送 失 败 ! ');history.back()</script>"; 

} 

}else{ 

$file = $_FILES[upload_file'[tmp_name'; 1/ 获取 附件 名 称 
$boundary = uniqid(""); /定义 分 界线 
$headers = "MIME-Version: 1.0 \n"; 

$headers .= "Content-type: multipart/mixed; Boundary= $boundary \An"; 


$headers .= "From:$mailfrom\An"; /定义 邮件 头 
if($_FILESrupload_ filelrtype]jf /判断 附 件 类 型 
$mimeType = $_FILES['upload file][type]; 

}else{ 

$mimeType ="application/unknown"; 1/ 获取 附件 类 型 

} 

$fileName = $_FILES['upload_fileJ[name'; // 获 得 附件 名 称 

$fp = fopen($file, "r"); /打开 文件 

$read = fread($fp, filesize($file)); /将 附件 读 入 变量 $read 
$read = base64_encode($read); /| 转换 为 base64 编码 
S$read = chunk_split($read); /将 长 字符 串 切 成 由 每 行 76 个 字符 组 成 的 小 块 
/* 定 义 邮 件 体 */ 


$body = "--$boundary— 

Content-type: text/plain; charset=iso-8859-1 

$mailbody 

--$boundary— 

Content-type: $mimeType; name=$fileName 
Content-disposition: attachment; filename=$fileName 
Content-transfer-encoding: base64 

S$read 

--$boundary—"; 

if(mail($mailto, $mailsubject, $body,$headers)}{ /用 mail() 函 数 发 送 邮件 
echo "<script>alert(' 附 件 发 送 成 功 ! ');history.back()</script>"; 
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}else{ 
echo "<script>alert(' 附 件 发 送 失败 ! ');history.back()</script>"; 
} 


3. 
?> 


a 
说明 当 所 发 送 的 邮件 中 带 有 附件 时 ， 首 先 需要 将 邮件 的 MIME 的 Content-type 类 型 定义 成 
multipart/mixed， 然 后 定义 邮件 内 容 分 割 线 ( 一 囊 字符 或 数字 )， 最 后 将 文件 内 容 编写 在 邮件 主体 内 
容 中 进行 发 送 。 


注意 pip 的 mail0 函 数 只 能 将 邮件 发 送 到 SMTP 的 邮件 转发 器 中 , 然后 通过 转发 器 将 邮件 发 送 
出 去 。 


12.5.3 ”接收 邮件 
接收 邮件 的 核心 是 调用 POP3 类 中 的 open0、login0、stat0 和 listmail0 方 法 , 实现 邮件 的 接收 操作 。 


POP3 类 独立 存储 于 mailclass.php 中 ， 将 在 12.6 节 中 进行 详细 讲解 。 这 里 将 具体 介绍 系统 是 如 何 调用 
POP3 类 实现 收 邮 件 功 能 的 ， 收 件 箱 页 面 如 图 12.18 所 示 。 


发 件 人 时间 大 小 


Xx 这 a Teg, 26 Dee 201 a 
元 寺 Med, es 2012 i 
X 小 可 受 jaofteorsoft cam St, Ee a .55K 
X 词 和 ER 中 下 1175K 
xX 国志 nerionireri con Sel D5 Jan 2013 a 
x 这 rsoft@nrsoft. con om oa 285.1 


O06:d4; 
您 的 邮箱 中 共有 也 件 : 6 封 占用 空间 : 1467K 


12.18” 收 件 箱 页 面 


在 receivemail.php 文件 中 ， 首 先 初始 化 Session 变量 ,判断 当前 用 户 是 否 具有 访问 权限 ， 然 后 通过 
include 语句 载 入 POP3 类 文件 mailclass.php; 最 后 实例 化 POP3 类 ， 通 过 返回 的 对 象 调用 类 中 的 方法 ， 
循环 输出 收 件 箱 中 的 邮件 信息 。 关 键 代 码 如 下 : 
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on 
<?php 

session_start(); 

if(lisset($_SESSION['user])X{ 

echo "<script>alert(' 请 不 要 非法 登录 本 站 ! ');history.back();</script>"; 


exit; 

} /以 上 代码 通过 判断 Session 变量 user 的 值 是 否 为 空 来 判断 用 户 是 否 非 法 登录 
include("mailclass.php"); // 包 含 发 邮件 类 

$rec=new pop3($_SESSION[host],110,10); /对 邮件 类 实例 化 

$rec->open(); // 调 用 邮件 类 的 open() 方 法 来 与 POP3 服务 器 建立 连接 


// 调 用 邮件 类 的 login() 方 法 来 验证 用 户 身份 

$rec-> login(substr($_SESSION['user],0,strpos($_SESSION['user],'@')),$_SESSION[pass"); 
S$rec->stat(); // 调 用 邮件 类 的 stat() 方 法 来 获得 邮件 总 数 和 邮件 总 字 节 数 
S$rec->listmail(); // 调 用 邮件 类 的 listmail() 方 法 获得 每 个 邮件 的 大 小 及 序号 


<?php 

if($rec->messages<=0X{ // 判 断 邮件 记录 数 

?> 

<table width="650" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 

<td bgcolor="#66CCFF"><div align="center"> 您 的 收 件 箱 内 无 邮件 ! </div></td> 

</tr> 

</table> 

<?php 

}else{ // 如 果 邮 件 存 在 ， 则 执行 下 面 的 操作 
?> 

<table width="650" border="0" align="center" cellpadding="0" cellspacing="0"> 

<tr> 

<td height="40” bgcolor="#66CCFF"> 

<table width="650" height="40" border="0" cellpadding="0" cellspacing="1"> 

<tr> 

<td height="20" colspan="2" bgcolor="E9E8E8"><div align="center"> 主 题 </div></td> 
<td width="136" bgcolor="E9E8E8"><div align="center"> 发 件 人 </div></td> 

<td width="183" bgcolor="E9E8E8"><div align="center"> 时 间 </div></td> 

<td width="95" bgcolor="E9E8E8"><div align="center"> 大 小 </div></td> 

</tr> 

<?php 

for($i=1;$i<=$rec->messages;$i++){ // 循 环 输出 邮件 数据 
$rec->getmail($i); 

?> 

<tr> 

<td width="43" height="20" bgcolor="E9E8E8"><div align="center> 

<a href="deletemail.php?del=<?php echo $i;?>"><img src="images/drop.png" border="0"></a> 
</div></td> 

<td width="187" bgcolor="E9E8E8"><div align="left"> 

<a href="readmail.php?mailid=<?php echo $i;?>"> 

<?php 

for($j=0;$j<count($rec->head);$j++\{ // 通 过 循环 提取 所 有 保存 在 head 数组 中 的 邮件 头 的 信息 
if(substr($rec->head[$j],0,20)=="Subject: =?GB2312?Q?"){  // 获 取 邮 件 主题 
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echo quoted_printable_decode(substr(htmlspecialchars($rec->head[$j]),20, 
strien(trim(htmlspecialchars($rec->head[$j])))-22))."<br>\n"; // 输 出 邮件 主题 

}else if(substr($rec->head[$j],0,7)=="Subject"X{ /去掉 邮件 主题 的 提示 文字 Subject 
echo base64_ decode(substr(htmlspecialchars($rec->head[$j]),8, 
strlen(trim(htmlspecialchars($rec->head[$j])))-8))."<br>\n"; 


} 

} 

?> 

</a> 

</div></td> 

<td height="20" bgcolor="E9E8E8"><div align="center"> 

<?php 

for($j=0;$j<count($rec->head);$j++X{ 

if(substr($rec->head[$j],0,4)=="From"}{ /获取 发 件 人 地 址 
// 去 掉 发 件 人 地 址 的 提示 文字 From 

echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec->head[$j])))-5)."<br>\n'" 
} 

} 

?> 

</div></td> 

<td height="20" bgcolor="E9E8E8"><div align="center> 

<?php 

for($j=0;$j<count($rec->head);$j++X{ // 获 取 头 文件 数据 
if(substr($rec->head[$j],0,4)=="Date"){ 1/ 截取 时 间 数 据 


echo substr(htmlspecialchars($rec->head[$)]),5,strlen(trim(htmlspecialchars($rec->head[$j])))-10)."<br>\n";// 输 
出 时 间 

下 

} 

?> 

</div></td> 

<td height="20" bgcolor="E9E8E8"><div align="center> 

<?php 

$size=$rec->mail_list[$i]['size']; 1/ 获取 文 件 大 小 
if($size>=1024X{ 1/ 计算 文件 大 小 ， 以 不 同 单位 进行 换算 输出 
echo substr(($size/1024),0,4)."M"; 

}elseif($size>1024*1024){ 

echo substr(($size/(1024*1024)),0,4)."G"; 

}elseif($size<1024X{ 

echo ($size)."KB"; 

} 

?> 

</div></td> 

</tr> 

<?php 

S$rec->head=NULL:; 

} 

?> 

</table> 

</td> 
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</tr> 

</table> 

<table width="650" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 

<td width="114"></td> 

<td width="536"><div align="right"> 您 的 邮箱 中 共有 邮件 :&nbsp;<?php echo $rec->messages;?>&nbsp; 封 占用 
空间 : 

<?php 

S$totalsize=$rec->size; 1/ 计算 邮件 服务 器 空间 大 小 
if($totalsize>=1024){ 

echo substr(($totalsize/1024),0,4)."M"; 

}elseif($totalsize>1024*1024){ 

echo substr(($totalsize/(1024*1024)),0,4)."G"; 

}elseif($totalsize<1024){ 

echo (S$totalsize)."KB"; 

} 

3 

&nbsp;</div></td> 

</tr> 

</table> 

<?php 

S$rec->close(); // 关 闭 服务 器 

3 


?> 


12.5.4 查看 邮件 详细 内 容 


在 收 件 箱 的 receivemail.php 文件 中 ， 单 击 邮 件 主题 超 链接 ， 将 跳 转 到 readmail.php 页 中 ， 在 该 文件 
中 根据 超 链 接 传递 的 邮件 id 值 ， 输 出 指定 邮件 的 详细 信息 。 其 仍然 是 调用 POP3 类 中 的 方法 ， 完 成 邮 
件 详细 信息 的 输出 操作 。 关 键 代码 如 下 : 


<?php 

session_start(); // 初 始 化 Session 变量 
ifllisset($_SESSIONruser])X // 判 断 用 户 的 权限 

echo "<script>alert(' 请 不 要 非法 登录 本 站 ! ');history.back();</script>"; 

exit; 

$mailid=$_GET[mailid]; // 获 取 超 链接 传递 的 邮件 ID 值 
include("mailclass.php"); // 载 入 POP3 类 

S$rec=new pop3($_SESSION[host],110,10); /实例 化 POP3 类 

$rec->open(); // 连 接 邮 件 服务 器 
S$rec->login(substr($_SESSION[user],0,strpos($_SESSION[user],'@')),$_SESSION['pass"]); // 用 户 登录 
$rec->stat(); /验证 用 户 是 否 正确 登录 

$rec->listmail(); // 读 取 邮 件 

S$rec->getmail($mailid); // 根 据 超 链 接 传 递 的 id 值 ， 获 取 指 定 邮件 的 内 容 
学 

<table width="650" border="0" align="center cellpadding="0" cellspacing="1"> 

<tr> 


_ 国 
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<td width="152" height="25" bgcolor="#CCCCCC"><div align="center"> 发 件 人 :</div></td> 

<td colspan="2" bgcolor="#E9E8E8"><div align="left"> 

<?php 

for($j=0;$j<count($rec->head);$j++X{ // 获 取 发 件 人 信息 
if(substr($rec->head[$j],0,4)=="From"){ /1/ 输 出 发 件 人 

echo substr(htmlspecialchars($rec->head[$)j]),5,stren(tim(htmlspecialchars($rec->head[$j])))-5)."<br>\n"; 
} 

} 

?> 

</div></td> 

</tr> 

<tr> 

<td height="25" bgcolor="#CCCCCC"><div align="center"> 收 件 人 :</div></td> 

<td height="20" colspan="2" bgcolor="#E9E8E8"><div align="left"><?php echo $_SESSION[user]; ?></div></td> 
</tr> 

<tr> 

<td height="25" bgcolor="#CCCCCC"><div align="center"> 邮 件 主 题 :</div></td> 

<td height="20" colspan="2" bgcolor="#E9E8E8"><div align="left"> 


<?php 

for($j=0;$j<count($rec->head);$j++\{ 1/ 获取 邮 件 主题 
if(substr($rec->head[$j],0,20)=="Subject: =?GB2312?Q?"X // 对 邮件 主题 进行 截取 
echo quoted_printable_decode(substr(htmlspecialchars($rec->head[$j]),20, 
strlen(trim(htmlspecialchars($rec->head[$j])))-22))."<br>vn"; /输出 邮件 主题 


}else if(substr($rec->head[$j],0,7)=="Subject"X{ 
echo base64_decode(substr(htmlspecialchars($rec->head[$j]),8， 
43 strlen(trim(htmlspecialchars($rec->head[$j])))-8))."<br>\n"; 
下 
} 
?> 
</div></td> 
</tr> 
<tr> 
<td height="25" bgcolor="#CCCCCC"><div align="center"> 发 件 时 间 :</div></td> 
<td height="20" colspan="2" bgcolor="#E9E8E8"><div align="left"> 
<?php 
for($j=0;$j<count($rec->head);$j++\{ // 获 取 发 件 时 间 
if(substr($rec->head[$j],0,4)=="Date")f{ // 对 发 件 时 间 进 行 截取 输出 
echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec->head[$j])))-10)."<br>\n'"; 
} 
} 
?> 
</div></td> 
</tr> 
<tr> 
<td height="25" bgcolor="#cccccc"><div align="center"> 邮 件 大 小 :</div></td> 
<td height="20" bgcolor="#E9E8E8"><div align="left"> 


<?php 
$size=$rec->mail_list[$mailid]['size']:; // 获 取 邮 件 大 小 
if($size>=1024){ // 对 邮件 大 小 进行 换算 输出 
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echo ($size/1024)."M"; 
}elseif($size>1024*1024){ 
echo ($size/(1024*1024))."G"; 
}elseif($size<1024X{ 
echo ($size)."KB"; 
} 
if(isset($rec->body[6])X{ 1/ 判断 是 否 存 在 附件 
S$file="; 
$filename=substr(strrchr($rec->body[6],"="),1,200); // 获 取 附 件 名 称 
for($i=0;$i<count($rec->body);$i++){ // 读 取 附 件数 据 
if($i>7 && $i<count($rec->body)-2X{ 
$file=base64_decode($rec->body[$i]); // 对 附件 进行 编码 转换 
} 
} 
file_put_contents("file/".$filename, $file); // 将 附件 数据 写 入 到 指定 文件 中 
} 
?> 
</div></td> 
<td height="20" bgcolor="#E9E8E8"><div align="center"> 
<input type="button" value=" 删 除 " class="buttoncss" 
‘onClick="javascriptwindow.location='deletemail.php?del=<?php echo $mailid;?>&filename=<?php echo $filename;?>"> 
</div></td> 
</tr> 
<?php 
if(isset($rec->body[6])X{ // 判 断 附件 是 否 存 在 , 存在 则 创建 附件 下 载 超 链接 
?> 
<tr> 
<td height="25" bgcolor="#cccccc"><div align="center"> 附 件 :</div></td> 
<td width="352" height="20" bgcolor="#E9E8E8"><?php echo' 
<a href="down.php?filename=".$filename." target=”blank">'.$filename.'</a>'; ?> 
</td> 
<td width="142" bgcolor="#E9E8E8">&nbsp;</td> 
</tr> 
<?php 
} 
?> 
<tr> 
<td height="325" colspan="3" bgcolor="#E9E8E8" valign="top"> 
<?php 
if(isset($rec->body[6])X{ 
echo base64_decode($rec->body[3])."<br>"; 
}else{ 
if(substr($rec->body[1],0,1)!="="X 
echo base64_decode($rec->body[1]); 
jelsef 
echo quoted_printable_decode($rec->body[1]); 
} 
} 


?> 
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</td> 
</tr> 
</table> 


12.5.5 下 载 附件 


下 载 附件 的 操作 从 查看 邮件 内 容 中 的 附件 超 链 接 开始 ,链接 到 down.php 文件 ， 在 该 文件 中 根据 超 
链接 传递 的 附件 名 称 实现 附件 下 载 的 操作 。 其 运行 结果 如 图 12.19 所 示 。 
网 


pls 生 1 月 ?日 | 
Jon, 07 Jan 2013 06:44:35 


285. 448242188 | 
网 要 01_ JPG 

EE 那里 有 漫山遍野 的 花 

文件 TF 戏 


mrsoftemrsoft. com 


mcofte@mraoft. com 


您 疏 打 开 或 保存 此 文件 四? 


名 称 ，fle 风味 01JPC 
加 区 型 ”kankan JPEG 图 入 
丰产 localhost 


> 


图 12.19 下 载 附 件 


在 down.php 文件 中 ， 根 据 超 链接 传递 的 附件 名 称 ， 通 过 header() 函 数 完成 附件 的 下 载 操 作 ， 其 关 
键 代 码 如 下 : 


<?phl 

eset // 初 始 化 Session 变量 
$filename='file/'.$_GET['filename']; // 获 取 附 件 在 服务 器 中 的 存储 位 置 
ifttisset($_SESSIONruser]) / 淹 断 当前 用 户 的 访问 权限 

echo "<script>alert( 对 不 起 ， 本 站 暂时 停止 该 文件 下 载 !");history.back();</script>"; 
exit; 

} 

S$fp=fopen($filename,"r"); 1! 打开 指定 的 文件 


header("Content-type:application/octet-stream"); 
header("Accept-ranges:bytes"); 
header("Accept-length:".filesize($filename)); 
header("Content-Disposition:attachment;filename=". $filename); 
echo fread($fp,filesize($filename)); /| 输出 文件 


@ 
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fclose($fp); /关闭 文件 
ww 


12.5.6 ”删除 邮件 


在 查看 邮件 详细 内 容 页 面 中 ， 单 击 “删除 ”按钮 ， 将 跳 转 到 deletemail.php 文件 中 ， 根 据 超 链接 传 
递 的 邮件 id 和 附件 名 称 ， 调 用 POP3 类 中 的 dele() 方 法 完成 邮件 的 删除 操作 。 如 果 存 在 附件 ， 则 应 用 
unlinkO) 函 数 删 除 存储 于 服务 器 下 的 附件 。deletemail.php 文件 的 代码 如 下 : 


| 一， 和 eeeeeseensseeesseseweseesseeesd 1GtGImiail Dh 


<?php 
session_start(); 
if(isset($_SESSION['user])){ // 判 断 当前 用 户 的 权限 


include("mailclass.php"); // 包 含 POP3 发 邮件 类 

S$rec=new pop3($_SESSION[host],110,10); // 对 类 进行 初始 化 

$rec->open(); /| 连接 POP3 服务 器 

// 对 当前 用 户 进行 身份 验证 
$rec->login(substr($_SESSION[usen],0,strpos($_SESSION[user],@'))$_SESSION['pass']); 
$rec->dele($_GET[del]); // 调 用 POP3 类 的 dele() 方 法 来 删除 $_GET[del] 指 定 的 邮件 
S$rec->close(); /关闭 与 POP3 服务 器 的 连接 ， 并 执行 删除 操作 


if(lempty($_GET['filename'"])X{ 
unlink(file/.$_GET[filename']); 


} 
echo "<script>alert(' 删 除 成 功 !');window.location.href='receivemail.php';</script>"; 


}else{ 
echo "<script>alert(' 您 不 具备 删除 权限 ， 请 正确 登录 !");window.location.href='index.php';;</script>"; 


; 
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12.5.7 ”查找 邮件 


为 了 方便 用 户 ， 该 邮件 系统 提供 了 两 种 搜索 方式 ， 分 别 为 按 邮 件 主 题 进行 查找 和 按 发 件 人 地 址 进 
行 查 找 。 查 找 邮 件 的 操作 在 findmail.php 文件 中 执行 ， 页 面 如 图 12.20 所 示 。 


条 理 拓 邮件 ，[ 硬 十 二 


12.20 ”查找 邮件 页 面 


查找 邮件 创建 的 form 表单 如 下 : 


<form name="form1" method="post" action="<?php echo $PHP_SELF;?>" onSubmit="return chkinput(this)"> 
<tr> 

<td width="143"><div align="right"> 查 找 邮 件 :</div></td> 

<td width="92"><div align="center"> 

<select name="findtype" class="inputcss"> 
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<option selected value=1> 邮 件 主题 </option> 

<option value=0> 发 件 人 </option> 

</select> 

</div></td> 

<td width="191"><div align="center"> 

<input type="text" name="info" size="30" class="inputcss" style="background-color:#e8f4ff " 
‘onMouseOver="this.style.backgroundColor='"#fffffr" onMouseOut="this.style.backgroundColor='#e8f4ff"> 
</div></td> 

<td width="224"><div align="left"> 

&nbsp;&nbsp;<input type="submit" value=" 开 始 " class="buttoncss" name="submit"> 
</div></td> 

</tr> 

</form> 


查找 邮件 的 处 理 操作 在 findmail.php 页 中 完成 ， 其 关键 代码 如 下 : 


nin Mail Dh errr 
<?php 

iflisset($_POST['submit]))f /判断 提交 按钮 值 是 否 被 设置 

?> 

<tr> 

<td width="188" height="20" bgcolor="#E9E8E8"><div align="center"> 主 题 </div></td> 
<td width="153" height="20" bgcolor="#E9E8E8"><div align="center"> 发 件 人 </div></td> 
<td width="209" height="20" bgcolor="#E9E8E8"><div align="center"> 时 间 </div></td> 
<td width="97" height="20" bgcolor="#E9E8E8"><div align="center"> 大 小 </div></td> 


</tr> 

<?php 

include("mailclass.php"); // 载 入 类 文件 

S$rec=new pop3($_SESSION[host],110,10); // 实 例 化 类 

$rec->open(); // 连 接 服务 器 
Srec->login(substr($_SESSION[user],0,strpos($_SESSION[user],'@')),$_SESSION['pass"]);// 用 户 登录 
S$rec->stat(); // 调 用 POP3 类 的 stat() 方 法 取得 总 邮件 数 和 总 邮件 大 小 
S$rec->listmail(); // 调 用 POP3 类 的 listmail() 方 法 取得 每 个 邮件 的 大 小 及 序号 
if($_POST['findtype]==1){ / 淹 断 查询 的 类 型 ，1 表示 根据 邮件 主题 查询 

$sum=0; 1// 用 $sum 记录 查找 到 邮件 的 总 个 数 
for($i=1;$i<=$rec->messages;$i++){ // 通 过 循环 遍历 所 有 邮件 

$rec->getmail($i); // 取 得 第 $i 封 邮 件 头 及 邮件 内 容 
for($j=0;$j<count($rec->head);$j++}{ // 通 过 循环 获取 邮件 头 所 有 内 容 
if(substr($rec->head[$j],0,7)=="Subject")f // 获 取 邮 件 主题 

// 去 掉 邮 件 主题 的 提示 文字 Subject: 


$head1=substr(htmlspecialchars($rec->head[$j]),8,strlen(trim(htmlspecialchars($rec->head[$j])))-8); 


if(preg_match(V$_POST[infolii",trim($head1)))f /利用 正则 表达 式 进行 匹配 查找 

$sum++; // 如 果 查 找到 匹配 的 主题 ， 则 使 sum 值 加 1 
?> 

<tr> 

<td height="20" bgcolor="#E9E8E8"><div align="left"> 

<a href="readmail.php?mailid=<?php echo $i;:?>"> 
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<?php 

for($j=0;$j<count($rec->head);$j++X{ // 显 示 邮 件 主 题 
if(substr($rec->head[$j],0,7)=="Subject")f 

echo substr(htmlspecialchars($rec->head[$)j]),8,stren(tim(htmlspecialchars($rec->head[$j])))-8)."<br>\n"; 
} 

} 

?> 

</a> 

</div></td> 

<td height="20" bgcolor="#E9E8E8"><div align="center"> 

<?php 

for($j=0;$j<count($rec->head);$j++X{ // 显 示 发 件 人 地 址 
if(substr($rec->head[$j],0,4)=="From'"){ 

echo substr(htmlspecialchars($rec->head[$)j]),5,stren(tim(htmlspecialchars($rec->head[$j])))-5)."<br>\n"; 
} 

} 

?> 

</div></td> 

<td height="20" bgcolor="#E9E8E8"><div align="center"> 

<?php 

for($j=0;$j<count($rec->head);$j++X{ // 显 示 发 件 时 间 
if(substr($rec->head[$j],0,4)=="Date"){ 

echo substr(htmlspecialchars($rec->head[$)]),5,stren(tim(htmlspecialchars($rec->head[$)])))-10)."<br>\n"; 
} 

] 

?> 

</div></td> 

<td height="20" bgcolor="#E9E8E8"><div align="center"> 

<?php 

$size=$rec->mail_list[$i['size']; // 显 示 邮 件 大 小 
if($size>=1024X 

echo substr(($size/1024),0,4)."M"; 

}elseif($size>1024*1024){ 

echo substr(($size/(1024*1024)),0,4)."G"; 

}elseif($size<1024X{ 

echo ($size)."KB"; 

} 

?> 

</div></td> 

</tr> 

<?php 

} 

S$rec->head=NULL:; 


} 

if($sum==0X{ // 如 果 $sum 的 值 最 终 还 为 0， 则 说 明 没有 查找 到 匹配 的 邮件 主题 ， 这 时 给 出 提示 
?> 

<tr> 

<td height="20" colspan="4" bgcolor="#E9E8E8"><div align="center"> 对 不 起 ， 没 有 查找 到 您 要 找 的 邮件 !</div> 
</td> 

</tr> 
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<?php 
} 
}elseif($_POST[findtype]==0){ /如 果 $_POST[findtype'] 的 值 为 0， 则 按 发 件 人 地 址 进行 查找 
$sum=0; 
for($i=1;$i<=$rec->messages;$i++){ 
$rec->getmail($i); 
for($j=0;$j<count($rec->head);$j++X{ 
if(substr($rec->head[$j],0,4)=="From'"){ 
$head1=substr(htmlspecialchars($rec->head[$j]),5,stren(trim(htmlspecialchars($rec->head[$j])))-5); 
} 
} 
if(trim($head1)==trim($_POST[info")X 
$sum++; 
?> 
<tr> 
<td height="20" bgcolor="#E9E8E8"><div align="left"><a href="readmail.php?mailid=<?php echo $i;?>"> 
<?php 
for($j=0;$j<count($rec->head);$j++){ 
if(substr($rec->head[$j],0,7)=="Subject")f 
echo substr(htmlspecialchars($rec->head[$j]),8,strlen(trim(htmlspecialchars($rec->head[$j])))-8)."<br>\n"; 
下 
} 
?> 
</a></div></td> 
<td height="20" bgcolor="#E9E8E8"><div align="center"> 
<?php 
for($j=0;$j<count($rec->head);$j++X{ 
if(substr($rec->head[$j],0,4)=="From"){ 
echo substr(htmlspecialchars($rec->head[$)]),5,stren(tim(htmlspecialchars($rec->head[$j])))-5)."<br>\n"; 
} 
} 
?> 
</div></td> 
<td height="20" bgcolor="#E9E8E8"><div align="center"> 
<?php 
for($j=0;$j<count($rec->head);$j++X{ 
if(substr($rec->head[$j],0,4)=="Date"){ 
echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec->head[$j])))-10)."<br>\n'"; 
} 
} 
?> 
</div></td> 
<td height="2" bgcolor="#E9E8E8"><div align="center> 
<?php 
$size=$rec->mail_list[$i]['size']; 
if($size>=1024X{ 

echo substr(($size/1024),0,4)."M"; 
}elseif($size>1024*1024X{ 

echo substr(($size/(1024*1024)),0,4)."G"; 
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}elseif($size<1024X{ 
echo ($size)."KB"; 
} 
?> 
</div></td> 
</tr> 
<?php 
} 
Srec->head=NULL:; 
} 
if($sum==0X{ 
?> 
<tr> 
<td height="20" colspan="4" bgcolor="#E9E8E8"><div align="center"> 对 不 起 ， 没 有 查找 到 您 要 找 的 邮件 !</div> 
</td> 
</tr> 
<?php 


12.6 注销 系统 


为 了 保护 个 人 隐私 ， 用 户 使 用 完 本 系统 后 应 及 时 退出 。 本 邮件 发 送 系统 完成 注销 系统 功能 的 文件 
为 logout.php， 其 详细 代码 如 下 : 


ltt OL. hanna 


<?php 

session_start(); /初始 化 Session 变量 
session_destroy(); /撤销 所 有 Session 变量 
header("location:index.php"); // 重 新 定位 到 登录 页 面 
?> 


注销 系统 模块 通过 session_destroy() 函 数 撤销 所 有 的 Session 变量 ， 这 样 保存 在 Session 变量 中 的 所 
有 与 用 户 有 关 的 信息 也 不 再 存在 。 


12.7 技术 提炼 


12.7.1 利用 mail() 函 数 发 送 电子 邮件 


PHP 中 提供 了 可 以 直接 发 送 电子 邮件 的 mail0 函 数 。 对 于 非 商业 用 的 邮件 系统 使 用 该 函数 最 为 方 
便 ， 由 于 mail0 函 数 不 能 进行 SMTP 认证 ， 所 以 最 好 安装 一 个 SMTP 服务 器 来 转发 邮件 。mail0O 函 数 的 
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语法 如 下 : 


bool mail(string to, string subject, string message[ ,string additional_headers][ ,string additional_parameters]) 


mail() 函 数 的 参数 说 明 如 表 12.1 所 示 。 


表 12.1 mail() 函 数 的 参数 说 明 


参数 描 述 
to 收 件 人 地 址 
Subject 电子 邮件 的 主题 
message 电子 邮件 的 内 容 


additional headers 


可 选 参数 ， 用 来 将 一 些 信息 插入 到 E-mail 的 头 部 ， 如 发 送 人 信息 、 抄 送 地 址 、 密 送 地 址 等 。 
该 参数 如 果 有 多 个 信息 ， 信 息 之 间 用 “nm ”分隔 


additional parameters 


其 他 邮件 参数 


全 as 注 总 用 mail(O 函 数 发 送 邮件 ， 需 要 在 php.ini 中 进行 SMTP 邮件 服务 器 地 址 和 端口 设置 ， 否 则 
邮件 无 法 通过 SMTP 服务 器 发 送出 去 。 设 置 如 下 : 


[mail function] 
; For Win32 only. 
SMTP = 192.168.1.42 


smtp_port = 25 
SMTP 是 发 送 邮件 的 服务 器 地 址 。smtp_port 为 SMTP 服务 器 的 端口 号 ， 一 般 默 认为 25。 


邮件 发 送 系统 是 通过 mailO 函 数 将 电子 邮件 发 送出 去 的 ， 所 以 应 该 在 系统 中 架设 一 个 SMTP 服务 


。 下 面 的 代码 为 该 邮件 发 送 系统 利用 mailO 函 数 发 送 邮件 的 过 程 。 


if(@mail($mailto, $mailsubject,$mailbody,$headers)) { // 通 过 mail() 函 数 发 送 邮件 ， 并 给 出 邮件 发 送 结果 
echo "<script>alert( 邮 件 发 送 成 功 ! ');history.back()</script>"; 
}else{ 
echo "<script>alert(' 邮 件 未 发 送 成 功 ! ');history.back()</script>"; 


一 


$mailto: 表示 收 件 人 地 址 。 

$mailsubject: 表示 邮件 主题 。 

$mailbody: 表示 邮件 内 容 。 

S$headers: 表示 邮件 头 内 容 。 

网 果 该 邮件 发 送 成 功 或 失败 系统 将 给 出 提示 ， 并 且 用 “@” 屏 蔽 错误 。 


回回 加 加 


12.7.2” 带 附件 的 邮件 发 送 


发 送 带 附件 的 邮件 并 不 是 简单 地 将 附件 内 容 上 传 到 POP3 服务 器 的 某 个 目录 下 ， 而 是 首先 将 附件 
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进行 二 进 制 编码 ， 之 后 将 该 编码 连接 到 邮件 内 容 后 面 ， 当 然 邮 件 内 容 和 附件 编码 之 间 需 要 用 标记 进行 
分 隔 ， 该 标记 在 邮件 头 中 定义 。 下 面 内 容 为 带 附件 的 邮件 组 织 形式 : 


Received: from mrkj ([192.168.1.227]) by MRZTS with Microsoft SMTPSVC(6.0.3790.0); 
Mon, 18 Dec 2006 09:02:40 +0800 
Date: Mon, 18 Dec 2006 00:57:26 +0000 
Subject: 12 
To: zts*™*@163.com 
Content-type: multipart/mixed; Boundary= 4585e7760018e 
From:zts*™*@163.com 
Return-Path: zts*™*@163.com 
Message-ID: 
X-OriginalArrivalTime: 18 Dec 2006 01:02:40.0187 (UTC) FILETIME=[372680B0:01C72240] 
--4585e7760018 
Content-type: text/plain; charset=iso-8859-1 
Content-transfer-encoding: 8bit 
(此 处 为 邮件 内 容 ) 
--4585e7760018 
Content-type: text/plain; name=shop.txt 
Content-disposition: attachment; filename=shop.txt 
Content-transfer-encoding: base64 
wffV4rXEtcS1xLXEtcS1xLTvtb21xNK7ICC13LXctcS1xA== 此 处 为 附件 的 base64 编码 ) 
--4585e7760018 


带 附 件 的 电子 邮件 的 不 同 部 分 之 间 用 分 界线 来 分 隔 ， 分 界线 在 Content--type 头 中 定义 ， 如 
Boundary= 4585e7760018e， 其 中 4585e7760018e 表示 不 同 部 分 的 分 界线 。 邮 件 的 每 个 新 部 分 以 两 个 连 
字号 〈--) 和 分 界线 开始 ， 最 后 一 个 分 界线 后 也 有 两 个 连 字 号 ， 表 示 该 邮件 中 没有 其 他 部 分 。 在 每 个 分 
界线 后 有 一 些 行 , 用 来 告诉 邮件 程序 这 部 分 内 容 的 类 型 , 如 上 面 例子 中 第 一 个 分 界线 后 面 的 以 Content- 
type:text/plain 开头 的 行 ， 这 些 行 说 明 后 面 的 部 分 是 ISO-8859-1 字符 集 的 纯 文本 形式 ， 跟 在 第 二 个 分 界 
线 后 的 行 告诉 邮件 程序 现在 的 部 分 是 一 个 .txt 文件 ， 文 件 名 是 shop.txt。Content-disposition 这 行 告诉 邮 
件 程序 如 果 可 能 就 以 内 和 嵌 的 方式 显示 附件 ， 现 在 新 的 邮件 程序 会 在 消息 后 显示 .txt 文件 的 内 容 ， 如 果 
Content-disposition 被 设 为 attachment， 那 么 邮件 程序 就 不 会 显示 .txt 文件 的 内 容 ， 而 是 显示 一 个 连接 到 
文件 的 图 标 , 收 件 人 要 看 附件 的 内 容 , 必须 单 击 这 个 图 标 。 一 般 情 况 下 , 如 果 附 件 是 一 些 文本 ，Content- 
disposition 会 被 设 为 inline， 这 是 因为 现在 大 部 分 邮件 程序 能 够 不 借助 其 他 浏览 器 而 直接 显示 附件 〈 文 
本 ) 的 内 容 。 如 果 附 件 不 是 文本 (如 图 片 或 其 他 类 似 的 内 容 ) ，Content-disposition 就 设 为 attachment。 


12.7.3 利用 fsockopen() 函 数 接收 电子 邮件 


上 面 介绍 的 mail0 函 数 只 能 发 送 邮 件 而 无 法 接收 , 利用 fockopen(O) 函 数 除了 可 以 发 送 邮件 外 , 还 可 
以 接收 邮件 。fsockopen(0) 函 数 的 语法 如 下 : 


int fsockopen( string hostname, int port [, int errno [, string errstr [, float timeout]]]) 


fsockopen() 函 数 的 参数 说 明 如 表 12.2 所 示 。 
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表 12.2 ”fsockopen() 函 数 的 参数 说 明 
参数 说 ”了 明 
hostname, 在 Internet 中 使 用 时 ,参数 hostname 及 po 分别 代表 网 址 及 埠 号 .在 UNIX 情形 下 可 做 IPC, hostname 
_port 
CITN 
Errst 


参数 表示 到 socket 的 路 径 ，port 配置 为 0 


2 主要 作为 错误 处 理 使 用 ， 可 以 省 略 
tr 


timeout 表示 在 指定 的 时 间 内 没有 连接 上 就 中 断 

利用 fsockopen() 函 数 连接 服务 器 后 ， 将 返回 一 个 socket 句柄 ， 之 后 通过 文件 操作 函数 ， 如 fgets()、 
fputs()、feoft) 等 接收 消息 或 发 送 命令 。 本 邮件 系统 将 利用 fsockopen() 函 数 接收 POP3 服务 器 邮箱 中 的 
内 容 。 


_ 
本 fsockopen0 函 数 可 以 与 很 多 服务 器 建立 连接 ， 如 SMTP 服务 器 、POP3 服务 器 等 。 
邮件 发 送 系统 利用 fsockopen() 函 数 连接 POP3 服务 器 ， 代 码 如 下 ， 


if(!I$this->connection=@fsockopen($this->hostname, $this->port,&$err_no,&$err_str, $this->timeout)){ 


S$this->hostname: 表示 POP3 服务 器 地 址 。 

S$this->port: 表示 POP3 服务 器 端口 号 。 

S$err_no: 表示 系统 连接 POP3 服务 器 发 生 错误 时 返回 的 错误 号 。 
$err_str: 表示 系统 连接 POP3 服务 器 发 生 错误 时 返回 的 错误 信息 。 
S$this->timeout: 表示 系统 连接 POP3 服务 器 时 最 大 超时 时 间 。 


12.7.4 封装 邮件 操作 类 


在 邮件 发 送 系 统 中 ， 将 收 邮 件 的 方法 封装 到 POP3 类 ， 并 且 将 其 单独 存储 于 mailclass.php 中 。 下 
面 对 收 邮件 类 POP3 进行 系统 讲解 。 
(1) 定义 收 邮件 类 中 的 数据 成 员 。 在 收 邮件 类 POP3 中 , 首先 定义 类 中 的 数据 成 员 ， 其 代码 如 下 : 


RPR 和 ER 
<!— mailclass.php: 一 > 


办 办 欠 欠 加 


class pop3{ 

public $hostname=""; /WPOP3 主机 名 

public $port=110; /主机 的 POP3 端口 号 ， 一 般 为 110 

public $timeout=5; /连接 主机 的 最 大 超时 时 间 

public $connection=0; /保存 与 主机 的 连接 

public $state="DISCONNECTED";”// 保 存 用 户 登录 状态 

public $debug=0; /标识 是 否 处 于 调试 状态 ， 如 果 是 则 输出 调试 信息 
public $err_str="; // 如 果 出 错 ， 保 存 错误 信息 

public $err_no="; // 如 果 出 错 ， 保 存 错误 号 

public $resp; /临时 保存 服务 器 的 响应 信息 

public $apop; /指示 需要 使 用 加 密 方式 进行 密码 验证 ， 一 般 服务 器 不 需要 
public $messages; // 保 存 邮箱 中 的 邮件 总 数 
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public $size; // 邮 箱 中 的 邮件 总 大 小 

public $mail_list; 1/ 一 维 数组 ， 保 存 各 个 邮件 的 大 小 及 其 在 POP 服务 器 上 的 序号 
public $head=array(); // 保 存 邮件 头 内 容 

public $body=array(); // 保 存 邮件 内 容 


(2) 定义 收 邮件 类 的 构造 函数 _construct)， 其 代码 如 下 : 
nallcla ss Php rrr errr 


function __construct($server, $port=110, $time_out=5){ 
S$this->hostname=$server; 

S$this->port=$port; 

S$this->timeout=$time_out; 

return true; 


} 


Wp 
当然 PHP 5 也 兼容 PHP 4 中 构造 函数 的 定义 方法 。 该 构造 函数 的 作用 是 为 数据 成 员 $hostname、$port、 
$timeout 初始 化 。 


(3) 定义 open() 方 法 连接 POP3 服务 器 。 在 open() 方 法 中 ， 通 过 fsockopen() 函 数 连接 POP3 服务 
器 ， 并 根据 返回 信息 判断 服务 器 是 否 连接 成 功 ， 如 果 成 功 ， 则 设置 标记 服务 器 的 数据 成 员 $state 的 值 为 
AUTHORIZATION。 代 码 如 下 : 


和 rrr 
<l— mailclass.php’ -> 


function open(){ 

if($this->hostname=="" 1/ 判断 服务 器 名 是 否 为 空 ， 是 则 给 出 提示 信息 
$this->err_str=" 请 输入 主机 名 !"; 

return false; 

} 

if($this->debug) 

echo "正在 连接 服务 器 !"; 

// 利 用 fsockopen() 连 接 POP3 服务 器 
if(!$this->connection=@fsockopen($this->hostname, $this->port,& $err_no,&$err_str, $this->timeout)){ 
$this->err_str=" 连 接 服务 器 失败 ， 错 误 信息 :".$err_str." 错 误 号 :".$err_no; 


return false; 

}else{ 

$this->getresp(); // 利 用 getresp() 方 法 处 理 服务 器 端 回 送 的 信息 
if($this->debug) // 显 示 返 回信 息 


S$this->outdebug($this->resp); 

if(substr($this->resp,0,3)!="+OK"X// 如 果 返 回信 息 的 前 3 个 字符 不 为 “+OK” 说 阴 服务 器 连接 失败 , 并 给 出 错误 信息 
$this->err_str=" 服 务 器 回 送 无 效 信息 ;".$this->resp." 请 检查 服务 器 是 否 正确 !"; 

return false; 

} 

S$this->state="AUTHORIZATION"; 

return true; 

} 

} 


@ 
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2 收 邮 件 类 判断 菜 条 命令 发 送 是 否 成 功 的 方法 是 : 通过 substr() 函 数 提取 出 返回 信息 的 前 3 
个 字符 , 如 果 该 字符 为 “tOK”, 表示 命令 被 成 功 执行 , 这 是 因为 如 果 POP3 中 某 条 命令 被 成 功 执行 ， 
返回 信息 的 前 3 个 字符 为 “+OK”。 下 面 是 侠 ockopen() 函 数 连 接 成 功 时 返回 的 信息 : 


+OK Microsoft Windows POP3 Service Version 1.0 <17073156@MRZTS> ready. 


(4) 使 用 getresp() 方 法 从 POP3 服务 器 接收 命令 。getresp() 方 法 用 于 取得 服务 器 端 回 送 的 信息 ， 并 
对 信息 做 如 下 处 理 : 去 掉 最 后 的 回 车 换行 符 后 ， 将 回 送信 息 保存 到 数据 成 员 $resp 中 。getresp0 方 法 的 
代码 如 下 : 


po yoyo ee 
<!| 一 mailclass.php: -> 


function getresp(X{ 

for($this->resp="";;X{ 

if(feof($this->connection)) 

return false; 

$this->resp.=fgets($this->connection,100); /用 fgets() 函 数 接收 命令 
$length=strlen($this->resp); /获取 返回 信息 的 长 度 
if($length>=2 && substr($this->resp,$length-2,2)=="\An"){ 
S$this->resp=strtok($this->resp,"\r\n"); /去 掉 回 车 换行 符 

return true; 

} 

} 

} 


(5) 使 用 outdebug0 方 法 对 特殊 字符 进行 转换 。outdebug() 方 法 的 作用 是 将 调试 信息 $messages 显 
示 出 来 ， 并 把 一 些 特殊 字符 进行 转换 及 在 行 尾 加 上 <br>， 这 样 可 使 输出 的 调试 信息 便于 阅读 和 分 析 。 
outdebug0 方 法 的 代码 如 下 : 


function outdebug($messageX{ 
echo htmlspecialchars($message)."<br>\n"; 


. 


(6) 使 用 command0 方 法 向 POP3 服务 器 发 送 命令 。command0 方 法 可 以 接收 以 下 3 个 参数 。 

$command: 发 送 给 服务 器 的 命令 。 

Sreturn_length: 指定 从 服务 器 回 送 信息 中 取得 多 长 作为 命令 回 送 标识 。 

S$return_code: 指定 标识 的 正确 值 。 

如 果 服 务 器 回 送 的 第 一 个 字符 为 “+”， 则 认为 命令 已 正确 执行 ， 也 可 以 用 前 面 提 到 的 “+OK” 来 
判断 。command() 方 法 的 代码 如 下 : 


rope pyro 
<!— mailclass.php: 一 > 


function command($command,S$retum_length=1,$retum_code="+'X{ 


if($this->connection==0X{ // 如 果 $this->connection 的 值 为 0， 说明 还 没有 连接 服务 器 
$this->err_str=" 没 有 连接 上 任何 服务 器 ， 请 检查 网 络 连接 !"; 
return false; 
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} 

if($this->debug) // 如 果 处 于 调试 状态 ， 则 给 出 调试 信息 
S$this->outdebug(">>> $command"); 

if(ffputs($this->connection,"$commandrwn")){ /如 果 发 送 命令 失败 ， 则 给 出 信息 
$this->err_str=" 无 法 发 送 命 令 ".$command; 

return false; 

}else{ 

S$this->getresp(); // 如 果 发 送 命令 成 功 ， 则 用 getresp() 方 法 来 接收 返回 的 信息 
if($this->debug) 

S$this->outdebug($this->resp); 

if(substr($this->resp,0,$return_length)!=$return_code){ // 如 果 返 回 的 信息 不 是 以 “+” 开 头 ， 说 阴 返 回 的 信息 无 效 
$this->err_str=$command." 命 令 ， 服 务 器 返回 无 效 :".$this->resp; 

return false; 

}else 

return true; 
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} 


(7) 使 用 login() 方 法 对 登录 到 POP3 服务 器 上 的 用 户 进行 身份 验证 。 如 果 验 证 通过 ， 则 设置 记录 
状态 的 数据 成 员 $state 的 值 为 TRANSACTION。 代 码 如 下 : 


nts nailclasS. hp rrr 
function login($user,$passwordX{ 

if($this->statel="AUTHORIZATION") /标志 状 态 的 值 不 为 AUTHORIZATION， 说 明 还 没有 连接 上 服务 器 
$this->err_str=" 还 没有 连接 到 服务 器 或 状态 不 对 !; 

return false; 

} 

if(!$this->apopX{ // 不 需要 加 密 方式 进行 密码 验证 

// 向 服务 器 发 送 USER 命令 ， 并 用 返回 信息 的 前 3 个 字符 是 否 为 “+OK” 来 验证 命令 是 否 发 送 成 功 
if(!$this->command("USER $user",3,"+OK")) 

return false; 

// 向 服务 器 发 送 PASS 命令 ， 并 用 返回 信息 的 前 3 个 字符 是 否 为 “+OK” 来 验证 命令 是 否 发 送 成 功 
if(l$this->command("PASS $password",3,"+OK")) 

return false; 

}else{ // 需 要 加 密 方 式 进行 密码 验证 

// 向 服务 器 发 送 APOP 命令 ， 以 加 密 的 方式 进行 身份 验证 

if(!$this->command("APOP $user".md5($this->greeting.$password),3,"+OK")) 

return false; 


} 
S$this->state="TRANSACTION"; 1/ 如果 验证 通过 ， 则 使 记录 状态 的 数据 成 员 $state 的 值 为 TRANSACTION 
return true; 


3 
用 户 成 功 登 录 服务 器 时 返回 的 信息 如 下 : 


+OK 
+OK User successfully logged on 


上 面 代 码 中 第 一 个 “+OK” 说 明 USER 命令 发 送 成 功 ， 第 二 个 “+OK” 说 明 PASS 命令 发 送 成 功 。 


辐 
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(8) stat() 方 法 获取 邮件 总 数量 及 总 大 小 。stat() 方 法 向 服务 器 发 送 STAT 命令 ， 从 而 获得 邮件 总 数 
量 和 总 大 小 。 代 码 如 下 : 


lmsmanensnnanarnnnnnssa naillciaSS. Dh "enna > 


function stat(){ 
if($this->state!="TRANSACTION"X // 判 断 用 户 是 否 已 经 成 功 登录 
$this->err_str=" 还 没有 连接 到 服务 器 或 没有 登录 成 功 !"; 


return false; 


} 

if(!$this->command("STAT",3,"+OK")) // 向 服务 器 发 送 STAT 命令 
return false; 
else{ 
S$this->resp=strtok($this->resp," "); /取得 “+OK” 
$this->messages=strtok(" "); /取得 邮件 总 数 
$this->size=strtok(" "); /取得 邮件 总 字 节 数 
return true; 
于 
} 


STAT 命令 发 送 成 功 后 返回 的 信息 如 下 : 


+OK 3 1504 


上 面 代码 中 ，+OK 说 明 命令 发 送 成 功 ，3 表示 邮箱 中 有 3 个 邮件 ; 1504 表示 邮件 总 大 小 为 1504 
字 节 。 
在 stat() 方 法 中 应 用 strtok0 函 数 对 字符 串 进行 分 隔 。strtok0 函 数 的 语法 如 下 : 


string strtok(string str1, string str2) 


上 述 代码 实现 了 从 strl 中 返回 str2 首 字 母 前 的 字符 串 ， 如 果 某 程序 中 有 多 个 strtok() 函 数 ， 只 需要 
第 一 个 strtok() 函 数 给 出 strl 的 值 ， 第 二 个 以 后 默认 的 strl 的 值 为 上 次 返回 值 后 剩余 的 字符 串 。 
(9) 定义 listmail0 方 法 获取 指定 邮件 的 大 小 ， 其 代码 如 下 : 


rope PEE 
<!— mailclass.php’ -> 


function listmail($mess=NULL,S$uni_id=NULL){ 


if($this->state!="TRANSACTION"){ /判断 用 户 是 否 已 经 成 功 登录 
$this->err_str=" 还 没有 连接 到 服务 器 或 没有 登录 成 功 "; 

return false; 

} 

if($uni_id) /如 果 指 定 $uni_id， 则 使 用 UIDL 命令 
$command="UIDL"; 

else 

$command="LIST"; // 如 果 没 指定 $uni_id， 则 使 用 LIST 命令 
if($mess) 

$command.=$mess; // 如 果 指 定 $mess， 则 使 用 $mess 所 指定 的 命令 
if(!$this->command($command,3,"+OK")}{ // 利 用 command() 方 法 发 送 命令 

return false; 

}else{ 


@ 
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$i=0; 

S$this->mail_list=array(); // 定 义 二 维 数组 ， 用 来 保存 每 个 邮件 的 大 小 
$this->getresp(); // 获 取 服 务 器 返回 的 信息 
while($this->resp!="."}{ 11“.” 为 邮件 结束 的 标志 

$itt; 


if($this->debug}{ 

S$this->outdebug($this->resp); 

} 

if($uni_idX{ 

S$this->mail_list[$iJ[num']=strtok($this->resp," "); 
Sthis->mail_list[$i]['size]=strtok(" "); // 获 取 某 个 邮件 的 大 小 

}else{ 

S$this->mail_list[$i["num"]=intval(strtok($this->resp," ")); 

S$this->mail_list[$i]["size"]=intval(strtok(" ")); 

} 

S$this->getresp(); 

} 

return true; 

} 

} 


(10) 定义 getmail() 方 法 获取 邮件 头 及 邮件 内 容 ， 代 码 如 下 : 


function getmail($num=1, $line=-1){ 


if($this->state!="TRANSACTION") /| 判断 用 户 是 否 已 经 成 功 登录 
$this->err_str=" 不 能 收 邮件 ， 还 没有 连接 服务 器 或 登录 成 功 !"; 

return false; 

1 /如 果 $line<0， 则 返回 所 有 的 邮件 内 容 
$command="RETR $num"; 

else // 如 果 $line>=0， 则 返回 邮件 前 $num 行内 容 


$command="TOP $num S$line"; 
if(!$this->command("$command",3,"+OK")) /向 服务 器 发 送 命令 
return false; 
else{ 
S$this->getresp(); // 如 果 命令 发 送 成 功 ， 则 获取 返回 信息 
$is_head=true; 
while($this->resp!="."X{ 
if($this->debug) 
S$this->outdebug($this->resp); 


if(substr($this->resp,0,1)==".") /1“.” 号 为 邮件 结束 的 标志 
$this->resp=substr($this->resp,1,strlen($this->resp)-1); 

if(trim($this->resp)=="") /邮件 头 与 邮件 内 容 以 空 行 分 隔 

$is_head=false; 

if($is_head) 

S$this->head[]=$this->resp; // 将 邮件 头 每 一 行 的 内 容 放 到 head[ 数 组 中 

else 

S$this->body[]=$this->resp; // 将 邮件 内 容 的 每 一 行 放 到 body[ 数 组 中 

$this->getresp(); 
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} 
return true; 
} 
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(11) 定义 dele() 方 法 删除 指定 邮件 ， 代 码 如 下 : 


function dele($numX{ 

if($this->statel="TRANSACTION"){ /判断 用 户 是 否 已 经 成 功 登录 
$this->err_str=" 不 能 删除 邮件 "; 

return false; 

} 

if(I$numX{ 

$this->err_str=" 删 除 的 邮件 参数 不 对 !"; 

return false; 


} 

if($this->command("DELE $num",3,"+OK")) /删除 Snum 指定 的 邮件 
return true; 

else 

return false; 


} 
(12) 定义 close() 方 法 断 开 与 服务 器 的 连接 ， 代 码 如 下 : 


function close(){ /close() 方 法 的 作用 是 删除 Snum 指定 的 邮件 
if($this->connection!=0){ 

if($this->state=="TRANSACTION") 

$this->command("QUIT",3,"+OK");，// 向 服务 器 发 送 QUIT 命令 ， 从 而 断 开 与 服务 器 的 连接 和 执行 DELE 命令 
fclose($this->connection); 

S$this->connection=NULL; 

S$this->state="DISCONNECTED"; 

} 

} 

} 


12.8 本 章 小 结 


本 章 Zend Framework 通过 Zend_Mail 组 件 构建 发 送 邮件 系统 ， 使 每 个 用 户 可 以 以 非常 快速 的 方式 
与 任何 一 个 角落 的 网 络 用 户 联络 ， 同 时 可 以 以 任何 方式 发 送 邮 件 。 充 分 利用 E-mail 功能 可 以 极 大 地 提 
高 工作 效率 ， 减 轻 用 户 的 工作 负担 。 通 过 本 章 的 学 习 ， 读 者 可 以 尝试 构建 自己 的 发 送 邮件 系统 。 


